存储过程不能处理代码

时间:2015-08-18 18:07:40

标签: c# sql .net stored-procedures

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

如果我通过任何方法(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;
    }
}

1 个答案:

答案 0 :(得分:0)

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

  

对于UPDATE,INSERT和DELETE语句,返回值为   受命令影响的行数。当a上存在触发器时   正在插入或更新的表,返回值包括数字   受插入或更新操作影响的行数和数字   受触发器或触发器影响的行数。对于所有其他类型的   语句,返回值为-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参数分配查询结果。 this Answer中解释了此更改的原因(否则您需要检查@Id参数中返回的NULL值)

如果您想阅读RETURN @Id,则需要在标有ParameterDirection.ReturnValue的查询中添加其他参数。

 cmd.Parameters.Add("@result", SqlDbType.Int).Direction = ParameterDirection.ReturnValue;
 ....

 int result = Convert.ToInt32(cmd.Parameters["@result"].Value.ToString());