- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在开始之前,我知道你不能从 UDF 调用存储过程,而且我知道这有各种“原因”(但对我来说没有多大意义,老实说,这听起来像是 Microsoft 的懒惰)部分)。
我更感兴趣的是如何设计一个系统来解决 SQL Server 中的这个缺陷。
这是我当前拥有的系统的快速概述:
我有一个动态报告生成器,用户可以在其中指定数据项、运算符(=、<、!= 等)和过滤器值。这些用于通过一个或多个过滤器构建“规则”,例如我的规则可能有两个过滤器“类别 < 12”和“位置!= 'York'”;
这些“规则”有成千上万条,其中一些有很多很多过滤器;
每个规则的输出都是一个法定报告,始终具有完全相同的“形状”,即相同的列/数据类型。基本上这些报告会生成吨位和 Material list ;
我有一个标量值函数,可以为指定规则生成动态 SQL,并将其作为 VARCHAR(MAX) 返回;
我有一个被调用来运行特定规则的存储过程,它调用 UDF 来生成动态 SQL,运行它并返回结果(这过去只返回结果,但现在我存储输出在进程键控表中,以使数据更易于共享,因此我返回此数据的句柄);
我有一个存储过程,调用它来运行特定公司的所有规则,因此它会列出要运行的规则列表,按顺序运行它们,然后将结果合并在一起作为输出。
所以这一切都很完美。
现在我想要最后一件事,一份运行公司摘要的报告,然后将成本应用于吨位/ Material 以生成成本报告。当我上周开始这样做时,这似乎是一个简单的要求:'(
我的报告必须是表值函数,才能与我已经编写的报告代理系统配合使用。如果我将其编写为存储过程,那么它将不会通过我的报告代理运行,这意味着它将不受控制,即我不知道谁运行了该报告以及何时运行。
但是我无法从表值函数中调用存储过程,处理此问题的两种明显方法如下:
获取 SQL 创建输出,运行它并获取结果。
--Method #1
WHILE @RuleIndex <= @MaxRuleIndex
BEGIN
DECLARE @DSFId UNIQUEIDENTIFIER;
SELECT @DSFId = [GUID] FROM NewGUID; --this has to be deterministic, it isn't but the compiler thinks it is and that's good enough :D
DECLARE @RuleId UNIQUEIDENTIFIER;
SELECT @RuleId = DSFRuleId FROM @DSFRules WHERE DSFRuleIndex = @RuleIndex;
DECLARE @SQL VARCHAR(MAX);
--Get the SQL
SELECT @SQL = DSF.DSFEngine(@ServiceId, @MemberId, @LocationId, @DSFYear, NULL, NULL, NULL, NULL, @DSFId, @RuleId);
--Run it
EXECUTE(@SQL);
--Copy the data out of the results table into our local copy
INSERT INTO
@DSFResults
SELECT
TableId, TableCode, TableName, RowId, RowName, LocationCode, LocationName, ProductCode, ProductName, PackagingGroupCode, PackagingGroupName, LevelName, WeightSource, Quantity, Paper, Glass, Aluminium, Steel, Plastic, Wood, Other, 0 AS General
FROM
DSF.DSFPackagingResults
WHERE
DSFId = @DSFId
AND RuleId = @RuleId;
SELECT @RuleIndex = @RuleIndex + 1;
END;
直接调用报告
--Method #2
WHILE @RuleIndex <= @MaxRuleIndex
BEGIN
DECLARE @DSFId UNIQUEIDENTIFIER;
SELECT @DSFId = [GUID] FROM NewGUID; --this has to be deterministic, it isn't but the compiler thinks it is :D
DECLARE @RuleId UNIQUEIDENTIFIER;
SELECT @RuleId = DSFRuleId FROM @DSFRules WHERE DSFRuleIndex = @RuleIndex;
DECLARE @SQL VARCHAR(MAX);
--Run the report
EXECUTE ExecuteDSFRule @ServiceId, @MemberId, @LocationId, @DSFYear, NULL, NULL, NULL, @RuleId, @DSFId, 2;
--Copy the data out of the results table into our local copy
INSERT INTO
@DSFResults
SELECT
TableId, TableCode, TableName, RowId, RowName, LocationCode, LocationName, ProductCode, ProductName, PackagingGroupCode, PackagingGroupName, LevelName, WeightSource, Quantity, Paper, Glass, Aluminium, Steel, Plastic, Wood, Other, 0 AS General
FROM
DSF.DSFPackagingResults
WHERE
DSFId = @DSFId
AND RuleId = @RuleId;
SELECT @RuleIndex = @RuleIndex + 1;
END;
我可以想到以下解决方法(没有一个特别令人满意):
在 CLR 中重写其中的一些内容(但这只是破坏规则的一大堆麻烦);
使用存储过程来生成我的报告(但这意味着我失去了对执行的控制,除非我为这个单一报告开发一个新系统,这与几十个都工作正常的现有报告不同);
将执行与报告分开,因此我有一个进程来执行报告,另一个进程只获取输出(但如果没有更多工作,就无法判断报告何时完成);
<等到 Microsoft 明白并允许从 UDF 执行存储过程。
还有其他想法吗?
<小时/>编辑于 2013 年 5 月 3 日,这是一个(非常)简单的示例,说明了如何将其结合在一起:
--Data to be reported
CREATE TABLE DataTable (
MemberId INT,
ProductId INT,
ProductSize VARCHAR(50),
Imported INT,
[Weight] NUMERIC(19,2));
INSERT INTO DataTable VALUES (1, 1, 'Large', 0, 5.4);
INSERT INTO DataTable VALUES (1, 2, 'Large', 1, 6.2);
INSERT INTO DataTable VALUES (1, 3, 'Medium', 0, 2.3);
INSERT INTO DataTable VALUES (1, 4, 'Small', 1, 1.9);
INSERT INTO DataTable VALUES (1, 5, 'Small', 0, 0.7);
INSERT INTO DataTable VALUES (1, 6, 'Small', 1, 1.2);
--Report Headers
CREATE TABLE ReportsTable (
ReportHandle INT,
ReportName VARCHAR(50));
INSERT INTO ReportsTable VALUES (1, 'Large Products');
INSERT INTO ReportsTable VALUES (2, 'Imported Small Products');
--Report Detail
CREATE TABLE ReportsDetail (
ReportHandle INT,
ReportDetailHandle INT,
DatabaseColumn VARCHAR(50),
DataType VARCHAR(50),
Operator VARCHAR(3),
FilterValue VARCHAR(50));
INSERT INTO ReportsDetail VALUES (1, 1, 'ProductSize', 'VARCHAR', '=', 'Large');
INSERT INTO ReportsDetail VALUES (2, 1, 'Imported', 'INT', '=', '1');
INSERT INTO ReportsDetail VALUES (2, 1, 'ProductSize', 'VARCHAR', '=', 'Small');
GO
CREATE FUNCTION GenerateReportSQL (
@ReportHandle INT)
RETURNS VARCHAR(MAX)
AS
BEGIN
DECLARE @SQL VARCHAR(MAX);
SELECT @SQL = 'SELECT SUM([Weight]) FROM DataTable WHERE 1=1 ';
DECLARE @Filters TABLE (
FilterIndex INT,
DatabaseColumn VARCHAR(50),
DataType VARCHAR(50),
Operator VARCHAR(3),
FilterValue VARCHAR(50));
INSERT INTO @Filters SELECT ROW_NUMBER() OVER (ORDER BY DatabaseColumn), DatabaseColumn, DataType, Operator, FilterValue FROM ReportsDetail WHERE ReportHandle = @ReportHandle;
DECLARE @FilterIndex INT = NULL;
SELECT TOP 1 @FilterIndex = FilterIndex FROM @Filters;
WHILE @FilterIndex IS NOT NULL
BEGIN
SELECT TOP 1 @SQL = @SQL + ' AND ' + DatabaseColumn + ' ' + Operator + ' ' + CASE WHEN DataType = 'VARCHAR' THEN '''' ELSE '' END + FilterValue + CASE WHEN DataType = 'VARCHAR' THEN '''' ELSE '' END FROM @Filters WHERE FilterIndex = @FilterIndex;
DELETE FROM @Filters WHERE FilterIndex = @FilterIndex;
SELECT @FilterIndex = NULL;
SELECT TOP 1 @FilterIndex = FilterIndex FROM @Filters;
END;
RETURN @SQL;
END;
GO
CREATE PROCEDURE ExecuteReport (
@ReportHandle INT)
AS
BEGIN
--Get the SQL
DECLARE @SQL VARCHAR(MAX);
SELECT @SQL = dbo.GenerateReportSQL(@ReportHandle);
EXECUTE (@SQL);
END;
GO
--Test
EXECUTE ExecuteReport 1;
EXECUTE ExecuteReport 2;
SELECT dbo.GenerateReportSQL(1);
SELECT dbo.GenerateReportSQL(2);
GO
--What I really want
CREATE FUNCTION RunReport (
@ReportHandle INT)
RETURNS @Results TABLE ([Weight] NUMERIC(19,2))
AS
BEGIN
INSERT INTO @Results EXECUTE ExecuteReport @ReportHandle;
RETURN;
END;
--Invalid use of a side-effecting operator 'INSERT EXEC' within a function
最佳答案
如果我处于你的情况,我不会尝试破解任何东西。我会像这样设置对象:
CREATE TABLE [dbo].[ReportCollection] (
[ReportCollectionID] int,
[ReportID] int
)
CREATE TABLE [dbo].[ReportResult] (
[ReportID] int,
[LocationCode] int,
[LocationName] nvarchar(max)
)
CREATE PROCEDURE [dbo].[usp_ExecuteReport] (
@ReportID int
)
AS
INSERT [dbo].[ReportResult]
SELECT @ReportID, 1, N'StackOverflow'
END
CREATE FUNCTION [dbo].[udf_RetrieveReportCollectionResults] (
@ReportCollectionID int
)
RETURNS @Results TABLE ([ReportID], [LocationCode], [LocationName])
AS
BEGIN
SELECT *
FROM [dbo].[ReportResult] rr
JOIN [dbo].[ReportCollection] rc
ON rr.ReportID = rc.ReportID
WHERE rc.ReportCollectionID = @ReportCollectionID
END
像这样使用它们:
INSERT [dbo].[ReportCollection] VALUES (1, 1)
INSERT [dbo].[ReportCollection] VALUES (1, 2)
EXEC [dbo].[usp_ExecuteReport] @ReportID = 1
EXEC [dbo].[usp_ExecuteReport] @ReportID = 2
SELECT * FROM [dbo].[udf_RetrieveReportCollectionResults](1)
每次运行报告时,都会开始一个新的集合。您的应用程序应该启动所有报告并随后整合结果。
--
如果您确实想要从 udf 调用存储过程(请不要这样做),请在 xp_cmdshell 上进行搜索。
关于sql-server - 如何避免从 SQL Server 中的 UDF 调用存储过程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16343930/
我在弄清楚如何从另一个 UDF 返回 UDF 中的数组时遇到了一些麻烦。这里的一个是简单的指数移动平均 UDF,我试图将数组返回到另一个 UDF,但我收到 #value 错误。我觉得有一个我没有看到的
我需要从另一个模块(在同一个工作簿中)调用以前制作的 UDF 来构建另一个 UDF。如何调用第一个函数? 这适用于 Excel VBA。我已经构建了我的第一个函数,它计算产品在特定时间的账面值(val
这个问题可能对许多 VBA 程序员有用。它涉及实现两个有用的独立任务并使它们同时工作。 第一个任务是为 UDF 制作 Excel 函数工具提示。虽然似乎还没有找到明确的解决方案,但目前我对自定义插入函
我正在将 Spark 与 Scala 一起使用,并希望将整行传递给 udf 并选择 udf 中的每个列名和列值。我怎样才能做到这一点? 我正在尝试以下 - inputDataDF.withColumn
这个问题在这里已经有了答案: Spark functions vs UDF performance? (3 个答案) 关闭2 年前。 我从 Pyspark 网站获取了以下 UDF,因为我试图了解是否
我已经使用 Spark 2.4 一段时间了,最近几天才开始切换到 Spark 3.0。切换到 Spark 3.0 运行后出现此错误 udf((x: Int) => x, IntegerType)
这个问题源自 SQLServer: Why avoid Table-Valued User Defined Functions? 。我开始在一些评论中提出问题,而对我评论的回复却偏离了主题。 这样您就
这是我的 hive 表 CREATE TABLE `dum`(`val` map>); insert into dum select map('A',array('1','2','3'),'B',ar
我想知道编写 spark udf 是否会降低性能。一般来说,我更喜欢组合做一件事的小函数…… 这是一个简单的例子,给定一个 DataFrame df: def inc = udf( (i: Doubl
我正在尝试根据另一列的值在 Spark 数据集中创建一个新列。另一列的值作为键在 json 文件中搜索,返回的值是用于新列的值。 这是我尝试过的代码,但它不起作用,而且我不确定 UDF 是如何工作的。
SPARK_VERSION = 2.2.0 我在尝试做 filter 时遇到了一个有趣的问题。在具有使用 UDF 添加的列的数据帧上。我能够用较小的数据集复制问题。 鉴于虚拟案例类: case cla
我正在 Java 中使用 Spark 来处理 XML 文件。来自databricks的spark-xml包用于将xml文件读入dataframe。 示例 xml 文件是: 1 joh
我正在尝试创建一个 MySQL UDF getFile(),它应该从磁盘上的某个目录返回文本文件的内容。问题是调用一次或两次有效,但在第二次或第三次调用 UDF 时,MySQL 服务器崩溃。 我无法通
我听说 Microsoft SQL Server 中有多种方法可以查找“最差”存储过程:按执行次数、按 CPU 工作时间、按队列等待时间等。 我正在寻找一种方法来查找最差(最慢/最常用)的 UDF -
我已经为一个项目构建了一个包含多个公式的 Excel 工作表。然后,我添加了一个用于折叠/展开某些单元组的命令按钮。 命令按钮代码是: Private Sub CommandButton1_Click
MySQL版本:5.1.73数据库客户端版本:libmysql - 5.1.73 我试图检查 NEW.src 在过去一小时内是否存在,如果不存在则执行 sys_exec udf。 我在 mysql 中
我正在尝试将元组列表传递给 scala 中的 udf。我不确定如何为此准确定义数据类型。我试图将它作为一整行传递,但它无法真正解决它。我需要根据元组的第一个元素对列表进行排序,然后返回 n 个元素。我
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 此问题是由拼写错误或无法再重现的问题引起的。虽然类似的问题可能是 on-topic
我正在尝试创建一个类似 =Extractinfo("A2","Name") 的函数,它可以从原始数据中提取姓名、电话和电子邮件 ID,一个用于所有 3 次提取的函数,我已经有一个提取电子邮件 ID 的
我正在编写一个用户定义函数(UDF),它以一些单元格作为参数。 这些单元格包含相同的数据,但精度不同;该功能显示可用的最佳精度。 函数的参数按精度升序编写。 这是一个例子: +---+--------
我是一名优秀的程序员,十分优秀!