访问SQL Server中的存储过程时出现错误
Server Error in '/' Application.
Procedure or function 'ColumnSeek' expects parameter '@template', which was not supplied.
当我通过.net的数据连接调用带有参数的存储过程到sql (System.data.SqlClient)
时,即使我提供参数,也会发生这种情况。这是我的代码。
SqlConnection sqlConn = new SqlConnection(connPath);
sqlConn.Open();
//METADATA RETRIEVAL
string sqlCommString = "QCApp.dbo.ColumnSeek";
SqlCommand metaDataComm = new SqlCommand(sqlCommString, sqlConn);
metaDataComm.CommandType = CommandType.StoredProcedure;
SqlParameter sp = metaDataComm.Parameters.Add("@template",SqlDbType.VarChar,50);
sp.Value = Template;
SqlDataReader metadr = metaDataComm.ExecuteReader();
我的存储过程是:
USE [QCApp]
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER PROCEDURE [dbo].[ColumnSeek]
@template varchar(50)
AS
EXEC('SELECT Column_Name, Data_Type
FROM [QCApp].[INFORMATION_SCHEMA].[COLUMNS]
WHERE TABLE_NAME = ' + @template);
我想弄清楚我在这里做错了什么。
编辑:事实证明,Template是null,因为我从通过URL传递的参数中获取了它的值,并且我搞砸了url参数传递(我使用的是@
代替&
)
答案 0 :(得分:270)
除了这里的其他答案,如果你忘记了:
cmd.CommandType = CommandType.StoredProcedure;
然后你也会收到这个错误。
答案 1 :(得分:78)
我会检查我的应用程序代码,看看你将@template设置为什么值。我怀疑它是空的,其中存在问题。
答案 2 :(得分:26)
这个问题确实通常是由于将参数值设置为null,因为上面提到的 HLGEM 。我想我会详细说明这个问题的一些解决方案,我发现这对这个问题的新用户有用。
我更喜欢的解决方案是将存储过程参数默认为NULL(或任何你想要的值),上面的 sangram 提到了这一点,但可能会错过,因为答案非常详细。有点像:
CREATE PROCEDURE GetEmployeeDetails
@DateOfBirth DATETIME = NULL,
@Surname VARCHAR(20),
@GenderCode INT = NULL,
AS
这意味着如果参数最终在某些条件下在代码中设置为null,则.NET不会设置参数,然后存储过程将使用它已定义的默认值。另一个解决方案,如果你真的想在代码中解决问题,那就是使用一种为你解决问题的扩展方法,例如:
public static SqlParameter AddParameter<T>(this SqlParameterCollection parameters, string parameterName, T value) where T : class
{
return value == null ? parameters.AddWithValue(parameterName, DBNull.Value) : parameters.AddWithValue(parameterName, value);
}
Matt Hamilton有一篇好的帖子here,在处理这个区域时列出了一些更好的扩展方法。
答案 3 :(得分:10)
我有一个问题,当我向整数参数提供0时,我会得到错误。并发现:
cmd.Parameters.AddWithValue("@Status", 0);
有效,但事实并非如此:
cmd.Parameters.Add(new SqlParameter("@Status", 0));
答案 4 :(得分:4)
我在调用存储过程时遇到了类似的问题
CREATE PROCEDURE UserPreference_Search
@UserPreferencesId int,
@SpecialOfferMails char(1),
@NewsLetters char(1),
@UserLoginId int,
@Currency varchar(50)
AS
DECLARE @QueryString nvarchar(4000)
SET @QueryString = 'SELECT UserPreferencesId,SpecialOfferMails,NewsLetters,UserLoginId,Currency FROM UserPreference'
IF(@UserPreferencesId IS NOT NULL)
BEGIN
SET @QueryString = @QueryString + ' WHERE UserPreferencesId = @DummyUserPreferencesId';
END
IF(@SpecialOfferMails IS NOT NULL)
BEGIN
SET @QueryString = @QueryString + ' WHERE SpecialOfferMails = @DummySpecialOfferMails';
END
IF(@NewsLetters IS NOT NULL)
BEGIN
SET @QueryString = @QueryString + ' WHERE NewsLetters = @DummyNewsLetters';
END
IF(@UserLoginId IS NOT NULL)
BEGIN
SET @QueryString = @QueryString + ' WHERE UserLoginId = @DummyUserLoginId';
END
IF(@Currency IS NOT NULL)
BEGIN
SET @QueryString = @QueryString + ' WHERE Currency = @DummyCurrency';
END
EXECUTE SP_EXECUTESQL @QueryString
,N'@DummyUserPreferencesId int, @DummySpecialOfferMails char(1), @DummyNewsLetters char(1), @DummyUserLoginId int, @DummyCurrency varchar(50)'
,@DummyUserPreferencesId=@UserPreferencesId
,@DummySpecialOfferMails=@SpecialOfferMails
,@DummyNewsLetters=@NewsLetters
,@DummyUserLoginId=@UserLoginId
,@DummyCurrency=@Currency;
动态构建搜索查询我在上面调用了一个:
public DataSet Search(int? AccessRightId, int? RoleId, int? ModuleId, char? CanAdd, char? CanEdit, char? CanDelete, DateTime? CreatedDatetime, DateTime? LastAccessDatetime, char? Deleted)
{
dbManager.ConnectionString = ConfigurationManager.ConnectionStrings["MSSQL"].ToString();
DataSet ds = new DataSet();
try
{
dbManager.Open();
dbManager.CreateParameters(9);
dbManager.AddParameters(0, "@AccessRightId", AccessRightId, ParameterDirection.Input);
dbManager.AddParameters(1, "@RoleId", RoleId, ParameterDirection.Input);
dbManager.AddParameters(2, "@ModuleId", ModuleId, ParameterDirection.Input);
dbManager.AddParameters(3, "@CanAdd", CanAdd, ParameterDirection.Input);
dbManager.AddParameters(4, "@CanEdit", CanEdit, ParameterDirection.Input);
dbManager.AddParameters(5, "@CanDelete", CanDelete, ParameterDirection.Input);
dbManager.AddParameters(6, "@CreatedDatetime", CreatedDatetime, ParameterDirection.Input);
dbManager.AddParameters(7, "@LastAccessDatetime", LastAccessDatetime, ParameterDirection.Input);
dbManager.AddParameters(8, "@Deleted", Deleted, ParameterDirection.Input);
ds = dbManager.ExecuteDataSet(CommandType.StoredProcedure, "AccessRight_Search");
return ds;
}
catch (Exception ex)
{
}
finally
{
dbManager.Dispose();
}
return ds;
}
然后在经过多次头部刮擦之后我将存储过程修改为:
ALTER PROCEDURE [dbo].[AccessRight_Search]
@AccessRightId int=null,
@RoleId int=null,
@ModuleId int=null,
@CanAdd char(1)=null,
@CanEdit char(1)=null,
@CanDelete char(1)=null,
@CreatedDatetime datetime=null,
@LastAccessDatetime datetime=null,
@Deleted char(1)=null
AS
DECLARE @QueryString nvarchar(4000)
DECLARE @HasWhere bit
SET @HasWhere=0
SET @QueryString = 'SELECT a.AccessRightId, a.RoleId,a.ModuleId, a.CanAdd, a.CanEdit, a.CanDelete, a.CreatedDatetime, a.LastAccessDatetime, a.Deleted, b.RoleName, c.ModuleName FROM AccessRight a, Role b, Module c WHERE a.RoleId = b.RoleId AND a.ModuleId = c.ModuleId'
SET @HasWhere=1;
IF(@AccessRightId IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.AccessRightId = @DummyAccessRightId';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.AccessRightId = @DummyAccessRightId';
END
IF(@RoleId IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.RoleId = @DummyRoleId';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.RoleId = @DummyRoleId';
END
IF(@ModuleId IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.ModuleId = @DummyModuleId';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.ModuleId = @DummyModuleId';
END
IF(@CanAdd IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.CanAdd = @DummyCanAdd';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.CanAdd = @DummyCanAdd';
END
IF(@CanEdit IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.CanEdit = @DummyCanEdit';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.CanEdit = @DummyCanEdit';
END
IF(@CanDelete IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.CanDelete = @DummyCanDelete';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.CanDelete = @DummyCanDelete';
END
IF(@CreatedDatetime IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.CreatedDatetime = @DummyCreatedDatetime';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.CreatedDatetime = @DummyCreatedDatetime';
END
IF(@LastAccessDatetime IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.LastAccessDatetime = @DummyLastAccessDatetime';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.LastAccessDatetime = @DummyLastAccessDatetime';
END
IF(@Deleted IS NOT NULL)
BEGIN
IF(@HasWhere=0)
BEGIN
SET @QueryString = @QueryString + ' WHERE a.Deleted = @DummyDeleted';
SET @HasWhere=1;
END
ELSE SET @QueryString = @QueryString + ' AND a.Deleted = @DummyDeleted';
END
PRINT @QueryString
EXECUTE SP_EXECUTESQL @QueryString
,N'@DummyAccessRightId int, @DummyRoleId int, @DummyModuleId int, @DummyCanAdd char(1), @DummyCanEdit char(1), @DummyCanDelete char(1), @DummyCreatedDatetime datetime, @DummyLastAccessDatetime datetime, @DummyDeleted char(1)'
,@DummyAccessRightId=@AccessRightId
,@DummyRoleId=@RoleId
,@DummyModuleId=@ModuleId
,@DummyCanAdd=@CanAdd
,@DummyCanEdit=@CanEdit
,@DummyCanDelete=@CanDelete
,@DummyCreatedDatetime=@CreatedDatetime
,@DummyLastAccessDatetime=@LastAccessDatetime
,@DummyDeleted=@Deleted;
这里我正在将存储过程的输入参数初始化为null,如下所示
@AccessRightId int=null,
@RoleId int=null,
@ModuleId int=null,
@CanAdd char(1)=null,
@CanEdit char(1)=null,
@CanDelete char(1)=null,
@CreatedDatetime datetime=null,
@LastAccessDatetime datetime=null,
@Deleted char(1)=null
为我做了诀窍。
我希望这对陷入类似陷阱的人有所帮助。
答案 5 :(得分:4)
对于我的情况,我必须从未定义DBNULL.Value
但值为null
的存储过程参数的代码中传递null
(使用if else条件)。
答案 6 :(得分:3)
如果未设置模板(即== null),也会引发此错误。
更多评论:
如果您在添加参数时知道参数值,也可以使用AddWithValue
不需要EXEC。您可以直接在SELECT中引用@template参数。
答案 7 :(得分:0)
首先 - 为什么是EXEC?不应该只是
AS
SELECT Column_Name, ...
FROM ...
WHERE TABLE_NAME = @template
目前的SP没有意义?特别是,这将寻找匹配@template的列,而不是@template的varchar值。即如果@template为'Column_Name'
,它将搜索WHERE TABLE_NAME = Column_Name
,这是非常罕见的(要使表和列的名称相同)。
此外,如果做必须使用动态SQL,则应使用EXEC sp_ExecuteSQL
(将值保留为参数)以防止注入攻击(而不是连接的连接)。但在这种情况下没有必要。
重新解决实际问题 - 一目了然看起来不错;你确定你没有SP的不同副本吗?这是一个常见错误......
答案 8 :(得分:0)
我遇到了同样的问题,要解决这个问题,只需在参数集中添加与存储过程完全相同的参数名称。
实施例
假设您创建了一个存储过程:
create procedure up_select_employe_by_ID
(@ID int)
as
select *
from employe_t
where employeID = @ID
因此,请确保完全按照存储过程中的参数命名
cmd.parameter.add("@ID", sqltype,size).value = @ID
如果你去
cmd.parameter.add("@employeID", sqltype,size).value = @employeid
然后发生错误。
答案 9 :(得分:0)
今天我遇到了这个错误,当空值传递给我的存储过程的参数时。我可以通过添加默认值= null来轻松修改存储过程。
答案 10 :(得分:0)
我遇到了同样的问题。我的参数为空值。所以我通过处理空值解决了。如果某人不确定运行时值并想处理null,则只需使用它。 (并且您不想更改SP /功能。)例如
stream[i]=sc.next();
答案 11 :(得分:0)
以防万一有人遇到,谷歌搜索。如果尽管定义了默认参数值(这应该允许您在不显式传递该参数的情况下运行该过程)仍然出现此错误,请仔细检查您是否确实设置了默认值,如下所示:
@someTextParameter NVARCHAR(100) NULL
只让参数接受 NULL
作为输入,而这个:
@someTextParameter NVARCHAR(100) = NULL
实际上是在声明参数可选。
这是一个愚蠢的错误,忽略了我花了半个小时;)
答案 12 :(得分:-1)
有必要告知正在调用存储过程:
comm.CommandType = CommandType.StoredProcedure;