存储过程不能处理代码



我有一个非常简单的存储过程,它返回给我一些输出变量。

如果我通过任何方法(Management Studio, Visual Studio Server Explorer)执行这个存储过程,它工作得很好。

问题是,如果我尝试在客户端使用ExecuteNonQuery()执行它,它不会执行存储过程,ExecuteNonQuery()返回值-1并且没有输出。我想知道我是否应该使用ExecuteNonQuery()以外的东西来执行这个过程,并得到这些输出变量。

这是我的存储过程:

CREATE PROCEDURE [dbo].ChekEmail
    @Email varchar(50),
    @Id int output,
    @Flag_User int output
AS
    SET @Id = 0
    SET @Id = (SELECT Client.Id FROM Client WHERE Client.Email = @Email)
    IF(@Id > 0)
    BEGIN 
        SET @Flag_User = 1
        RETURN @Id
    END
    ELSE
    BEGIN 
        SET @Id = (SELECT Clinic.Id FROM Clinic WHERE Clinic.Email = @Email)
        IF(@Id > 0)
        BEGIN 
            SET @Flag_User = 2
            RETURN @Id
        END
    END

在这里执行

public int CheckEmail(string Email)
{
    Conection con = new Conection();
    SqlCommand cmd = new SqlCommand("ChekEmail");
    cmd.Connection = con.connect();
    cmd.Parameters.AddWithValue("@Email", Email);
    cmd.Parameters.Add("@Id", SqlDbType.Int).Direction = ParameterDirection.Output;
    cmd.Parameters.Add("@Flag_User", SqlDbType.Int).Direction = ParameterDirection.Output;
    cmd.CommandType = CommandType.StoredProcedure;
    int x = cmd.ExecuteNonQuery();
    if (x >= 0)
    {
        int Id_Owner = Convert.ToInt32(cmd.Parameters["@Id"].Value.ToString());
        int Flag_User = Convert.ToInt32(cmd.Parameters["@Flag_User"].Value.ToString());
        if (Flag_User == 1)
        {
            return Id_Owner;
        }
        else
        {
            return 0;
        }
    }
    else
    {
        return 0;
    }
}

你可以在MSDN页面上看到ExecuteNonQuery为SELECT查询返回-1是一个记录的行为

对于UPDATE、INSERT和DELETE语句,返回值是受命令影响的行数。当一个触发器存在于表被插入或更新时,返回值包含数字受插入或更新操作和编号影响的行受一个或多个触发器影响的行。对于所有其他类型的语句,返回值为-1。如果发生回滚,则返回值也是-1

但是这并不意味着您的存储过程没有被执行。简单地说,当前存储过程中没有返回受影响的行,因为您没有使用任何必需的命令,而只是使用SELECT。
不要将RETURN @Id误认为ExecuteNonQuery返回的值。您只需在ExecuteNonQuery调用

之后删除if

也可以将存储过程更改为

CREATE PROCEDURE [dbo].ChekEmail
    @Email varchar(50),
    @Id int output,
    @Flag_User int output
AS
    SET @Id = 0
    SELECT @Id = Client.Id FROM Client WHERE Client.Email = @Email
    IF(@Id >0 )
    BEGIN 
        SET @Flag_User = 1
        RETURN @Id
    END
    ELSE
    BEGIN 
        SELECT @Id = Clinic.Id FROM Clinic WHERE Clinic.Email = @Email
        IF(@Id >0 )
        BEGIN 
            SET @Flag_User = 2
            RETURN @Id
        END
    END

在这个更新后的存储过程中,我使用非标准SELECT将查询结果赋给@ID参数。此更改的原因在此答案中有很好的解释(否则您需要检查@Id参数中返回的NULL值)

如果你想读取RETURN @Id,你需要在你的查询中添加另一个参数,标记为ParameterDirection.ReturnValue

 cmd.Parameters.Add("@result", SqlDbType.Int).Direction = ParameterDirection.ReturnValue;
 ....
 int result = Convert.ToInt32(cmd.Parameters["@result"].Value.ToString());

最新更新