1

我的 orm 工具执行程序的方式有问题。

这就是我执行的方式:

exec PostViewProc @Id='18767F6A-FF7A-47E9-AF09-6DB8A2F3B20E',@AuthorId='5455D9B9-B25A-41BD-BD2C-C9CBAE87D629'

按预期返回 1 行..

这就是 orm 工具生成它的方式:

exec sp_executesql 
N'EXEC [PostViewProc] @Id, @AuthorId',
N'@Id uniqueidentifier,@AuthorId uniqueidentifier',
@Id='18767F6A-FF7A-47E9-AF09-6DB8A2F3B20E',@AuthorId='5455D9B9-B25A-41BD-BD2C-C9CBAE87D629'

什么都没有!

第一个工作正常。但是 orm 生成的“sp_executesql”永远无法正常工作。如果我删除@AuthorId,orm 生成的可以正常工作,但是我添加了一些其他参数,它又会崩溃。

我还有一些其他的程序,一遍又一遍地生活在同样的问题上。不同的参数会导致意想不到的结果,即使是参数的数量也会导致意想不到的结果。甚至错误。(使用 sp_executesql 时,其他的工作都很好)

我需要对这里发生的事情进行一些很好的解释,因为程序本身没有任何问题,我现在很困惑。

4

3 回答 3

0

我们确实讨论了这种情况,并决定最好的解决方案是以精确的顺​​序传递所有参数。如果该过程为其自己的参数处理空值,则只需将它们设置为空或任何其他最适合的值。

感谢你们的时间、耐心和兴趣。@GarethD & @Mukund

于 2015-02-17T10:25:34.833 回答
0

好的,正如评论中所指出的,问题是参数的顺序。为了演示这个问题,想象以下简单的过程

CREATE PROCEDURE dbo.TestProc @A INT = NULL, @B INT = NULL, @C INT = NULL
AS
BEGIN
    SELECT TOP 1 A = @A, B = @B, C = @C;
END
GO

要记住的关键是变量和参数之间没有映射,例如

DECLARE @C INT = 1;
EXECUTE dbo.TestProc @C;

无法识别您的变量已被调用,@C因此您希望将其分配给同名的参数。如果在调用过程时没有显式地为参数分配值,那么它们将按照先到先得的原则分配,因此上面将返回:

A   B       C
------------------
1   NULL    NULL

实际上,由于您的变量只不过是一个占位符,因此上述查询与仅执行没有什么不同:

EXECUTE dbo.TestProc 1;

您当然不会期望程序知道您打算将 1 作为 @C 而不是 @A 的值传递。您的 ORM 生成的查询有类似的问题。与上述过程等效的代码是:

EXECUTE sp_executesql N'EXEC dbo.TestProc @A, @C', N'@A INT, @C INT', @A = 1, @C = 2;

返回:

A   B   C
------------
1   2   NULL

现在在评论中回答你的一些问题。

我在使用 sp_executesql 时确实传递了参数名称和类型,不是吗?

是的,您可以,但这不是强制性的,以下两个语句将产生相同的结果:

EXECUTE sp_executesql 
    N'EXEC dbo.TestProc @A = @A, @B = @B, @C = @C', 
    N'@A INT, @B INT, @C INT', 
    @A = 1, @B = 2, @C = 3; 

EXECUTE sp_executesql 
    N'EXEC dbo.TestProc @A, @B, @C', 
    N'@A INT, @B INT, @C INT', 
    1, 2, 3; 

如果您只想传递某些值,那么建议使用命名参数,但不是强制性的。不同之处在于,如果您不使用参数名称,则必须以正确的顺序提供参数:

EXECUTE sp_executesql 
    N'EXEC dbo.TestProc @A = @A, @C = @C', 
    N'@A INT, @C INT', 
    @A = 1, @C = 3; 

EXECUTE sp_executesql 
    N'EXEC dbo.TestProc @A, @B, @C', 
    N'@A INT, @B INT, @C INT', 
    1, NULL, 3; 

为什么proc无法匹配它们?

如上所述,该过程不知道您的变量名称,只知道它的值,因此以下两个语句是等效的:

DECLARE @C INT = 1;
EXECUTE dbo.TestProc @c;

EXECUTE dbo.TestProc 1;

如果您检查前一个查询的执行计划 XML,您可以确认这一点:

<ParameterList>
    <ColumnReference Column="@C" ParameterCompiledValue="NULL" ParameterRuntimeValue="NULL" />
    <ColumnReference Column="@B" ParameterCompiledValue="NULL" ParameterRuntimeValue="NULL" />
    <ColumnReference Column="@A" ParameterCompiledValue="NULL" ParameterRuntimeValue="(1)" />
</ParameterList>

对传递给过程的变量名称的任何引用都将丢失。此外,即使它没有丢失,也没有理由匹配它,如果不匹配,你会如何处理它,例如

EXECUTE dbo.TestProc @D, @A, 2;

@A匹配参数名称,但@D不匹配,常量也不匹配2。没有明智的方法来解决这个问题,而且 AFIK 没有语言可以从传递的变量的名称中推断出参数。

这 4 行 tsql 代码是做什么的?

1.  exec sp_executesql 
2.  N'EXEC [PostViewProc] @Id, @AuthorId',
3.  N'@Id uniqueidentifier,@AuthorId uniqueidentifier',
4.  @Id='18767F6A-FF7A-47E9-AF09-6DB8A2F3B20E',@AuthorId='5455D9B9-B25A-41BD-BD2C-C9CBAE87D629'
  • 第 1 行只是过程的调用sp_executesql
  • 第 2 行是语句参数 - 这是要执行的 sql。
  • 第 3 行是@params参数,这包含了语句中使用的所有参数的定义。
  • 第 4 行是 中使用的参数的值@params,除非显式分配(例如@A = 1, @B = 2),否则这些参数的分配顺序与定义参数的顺序相同

我还应该对该 proc 定义什么以使其了解这里发生的情况?

我不知道您使用的是什么 ORM,所以我不确定如何强制它生成正确的代码,它应该生成:

EXEC sp_executesql 
N'EXEC [PostViewProc] @Id = @Id, @AuthorID = @AuthorId',
N'@Id uniqueidentifier,@AuthorId uniqueidentifier',
@Id='18767F6A-FF7A-47E9-AF09-6DB8A2F3B20E',@AuthorId='5455D9B9-B25A-41BD-BD2C-C9CBAE87D629';

第二行的区别已经从

N'EXEC [PostViewProc] @Id, @AuthorId',

N'EXEC [PostViewProc] @Id = @Id, @AuthorID = @AuthorId',

如果您不能强制使用命名参数,那么您必须以正确的顺序传递所有值,包括 NULL 值。

于 2015-02-17T10:48:37.213 回答
-1

尝试以下脚本,

exec sp_executesql '18767F6A-FF7A-47E9-AF09-6DB8A2F3B20E','5455D9B9-B25A-41BD-BD2C-C9CBAE87D629'
于 2015-02-17T09:27:40.477 回答