我从C#ASP.Net应用程序调用SQL函数(UDF)。我调用的函数需要一个类型为“uniqueidentifier”的参数。
如果我在SqlCommand的CommandText中进行调用并传递“null”,结果会在大约3秒内返回...
SqlCommand Command = new SqlCommand();
Command.Connection = (SqlConnection)(DbDataModifier.CreateConnection());
Command.CommandText = "select * from GetLocalFirmLoginsSummary(null) order by Date asc"
Command.CommandType = CommandType.Text;
Command.Connection.Open();
SqlDataReader Reader = Command.ExecuteReader(); // takes 3 seconds
但是,如果我进行调用并将DBNull.Value作为SqlParameter传递,结果需要60秒才能返回...
SqlCommand Command = new SqlCommand();
Command.Connection = (SqlConnection)(DbDataModifier.CreateConnection());
Command.CommandText = "select * from GetLocalFirmLoginsSummary(@CustomerGroupID) order by Date asc"
Command.CommandType = CommandType.Text;
SqlParameter Param = new SqlParameter();
Param.ParameterName = "@CustomerGroupID";
Param.SqlDbType = SqlDbType.UniqueIdentifier;
Param.Direction = ParameterDirection.Input;
Param.IsNullable = true;
Param.Value = DBNull.Value;
Command.Parameters.Add(Param);
Command.Connection.Open();
SqlDataReader Reader = Command.ExecuteReader(); // takes over 60 seconds
如果我在SQL Management Studio中运行相同的查询,大约需要3秒,即使将null作为参数传递...
declare @CustomerGroupID uniqueidentifier
set @CustomerGroupID = null
select * from GetLocalFirmLoginsSummary(@CustomerGroupID)
我正在调用的函数定义为:
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER FUNCTION [dbo].[GetLocalFirmLoginsSummary]
(
@CustomerGroupID uniqueidentifier
)
RETURNS TABLE
AS
RETURN
(
SELECT CustomerGroupID, CustomerGroupName, USR, DATEADD(MONTH, DATEDIFF(MONTH, 0, createdDate), 0) AS Date, COUNT(*) AS Quantity
FROM dbo.GetLocalFirmLogins(@CustomerGroupID) AS Logins
GROUP BY USR, CustomerGroupID, CustomerGroupName, DATEADD(MONTH, DATEDIFF(MONTH, 0, createdDate), 0)
)
...
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER FUNCTION [dbo].[GetLocalFirmLogins]
(
@CustomerGroupID uniqueidentifier
)
RETURNS TABLE
AS
RETURN
(
SELECT vw_CustomerGroupAccountDetails.CustomerGroupID, vw_CustomerGroupAccountDetails.CustomerGroupName, Organisations.USR, Organisations.town AS FirmTown,
Users.id AS ID, Users.userName, Users.password, Users.isPartner, Users.isFeeEarner, Users.nlisUserName, Users.nlisPassword, Users.email, Users.lastLoginDate, Users.createdDate
FROM vw_CustomerGroupAccountDetails RIGHT OUTER JOIN
Organisations ON vw_CustomerGroupAccountDetails.AccountID = CAST(Organisations.id AS nvarchar(50)) RIGHT OUTER JOIN
Users ON Organisations.clientId = Users.uploadedBy
WHERE (Users.canLogin = 1) AND (Users.isDeleted = 0) AND (NOT (Organisations.USR IS NULL)) AND
((vw_CustomerGroupAccountDetails.CustomerGroupID = @CustomerGroupID) OR (@CustomerGroupID IS NULL))
)
那么当我使用SqlCommand的SqlParame从C#调用SQL函数时,它会花多长时间呢?
我正在使用SQL Server 2000。
我到处搜索并以各种方式进行搜索,但找不到其他人有同样的问题。
答案 0 :(得分:1)
吉姆,你能发布dbo.GetLocalFirmLogins的来源了吗?
正如马丁所说,这可能是参数嗅探。如果是,在代码中的查询提示,其中寻找@CustomerGroupID可能是"手术"解决这个问题。
在dbo.GetLocalFirmLogins中为查询添加OPTION(LOOP JOIN)可以在这里修复。
我想我也可以解释为什么SSMS没有看到问题,而C#就是这样。这部分查询:
(CustomerGroupID = @CustomerGroupID) OR (@CustomerGroupID IS NULL)
可能有两个不同的计划,具体取决于您来自的连接的ANSI NULL设置。 ANSI NULL和效果在此处描述:
您可以通过检索此查询的结果来查看连接的ANSI设置:
SSMS可能具有与C#编码不同的NULL设置,并导致对查询的不同评估。
答案 1 :(得分:1)
答案 2 :(得分:0)
Management Studio和.NET代码之间的性能差异通常与ANSI_NULLS等设置有关。
这可能是SQL Server 2000中的一个错误 - 请参阅this post,这似乎暗示在不考虑CREATE FUNCTION之前SET ANSI_NULLS ON。
我没有SQL Server 2000来测试这个,但也许你可以尝试编写你的函数脚本。