从存储过程获取输出参数的替代方法
Alternative way to get output parameter from stored procedure
我喜欢使用 Dapper 来满足我的 ORM 需求,但我知道必须有更好的方法 insert/update 我的 sql 服务器数据库使用存储过程和强类型列表。
例如:
我有一首class歌曲:
public class Song
{
public int Id { get; set; }
public string title { get; set; }
public string genre { get; set; }
}
有人提交了歌曲列表:
List<Song> songs = new List<Song> {
new Song { Id = 1, title = "Song 1" , genre="rock"},
new Song { Id = 2, title = "Song 2" , genre="disco"}};
我想使用我的存储过程更新数据库,该过程可以插入新歌曲或在歌曲已存在时更新它。我的存储过程有两个输出参数:
@success_added int = 0 and @success_updated int = 0
我的存储过程如下:
ALTER PROCEDURE [dbo].[UpdateSong]
@Id int = 0,
@title varchar(25) = NULL,
@genre varchar(25) = NULL,
@success_updated bit = 0 OUTPUT,
@success_added bit = 0 OUTPUT
AS
IF NOT EXISTS (SELECT Id FROM Songs WHERE Id = @Id)
BEGIN
INSERT INTO Songs
(
-- Id created by db
title,
genre
)
VALUES
(
@title,
@genre
)
SELECT @Success_Added = 1, @Success_Updated = 0
END
ELSE -- song already exists
BEGIN
UPDATE Songs
SET
title = @title,
@genre = @genre
WHERE Id = @Id
SELECT @Success_Added = 0, @Success_Updated = 1
END
RETURN
我知道这行得通:
dbConn.Open();
DynamicParameters p = new DynamicParameters();
foreach (var song in songs)
{
p.Add("@Id", song.Id);
p.Add("@title", song.title);
p.Add("@genre", song.genre);
p.Add("@success_updated", dbType: DbType.Boolean, direction: ParameterDirection.Output);
p.Add("@success_added", dbType: DbType.Boolean, direction: ParameterDirection.Output);
dbConn.Execute("Test_UpdateSong", p, commandType: CommandType.StoredProcedure);
Console.WriteLine("@success_added: " + p.Get<Boolean>("@success_added"));
Console.WriteLine("@success_updated: " + p.Get<Boolean>("@success_updated"));
}
dbConn.Close();
但这需要手动将每首歌曲 属性 转换为匿名类型 DynamicParameter。我宁愿简单地这样做:
dbConn.Open();
foreach (var song in songs)
{
var updateResult = dbConn.Query<dynamic>("Test_UpdateSong", song, commandType: CommandType.StoredProcedure);
}
dbConn.Close();
这也行得通。但是,现在如何获取我的输出参数?
正如我最初所说,我不想手动将每个 class 属性 转换为 Dapper 动态参数。这很关键,因为如果我创建一个泛型方法,我可能不知道哪个 class 被传递到该方法中,因此不知道要传递哪些属性以转换为动态参数。采纳@Metro Smurfs 的建议(如果其他方法都失败了,请阅读说明),我查看了 Dapper 测试 class 并找到了适合我的解决方案:
DynamicParameters p = new DynamicParameters(song);
通过将歌曲对象添加到 DynamicParameters 构造函数,创建了一个 DynamicParameters 模板,它会自动将所有属性转换为参数。然后我可以简单地添加我的两个输出参数并执行 sproc:
p.Add("@success_updated", dbType: DbType.Boolean, direction: ParameterDirection.Output);
p.Add("@success_added", dbType: DbType.Boolean, direction: ParameterDirection.Output);
dbConn.Execute("Test_UpdateSong", p, commandType: CommandType.StoredProcedure);
// get my output parameters...
var success_added = p.Get<bool>("@success_Added");
var success_added = p.Get<bool>("@success_Updated");
我准备好了!感谢@Nick 和@Metro Smurf 的建议!
我喜欢使用 Dapper 来满足我的 ORM 需求,但我知道必须有更好的方法 insert/update 我的 sql 服务器数据库使用存储过程和强类型列表。
例如:
我有一首class歌曲:
public class Song
{
public int Id { get; set; }
public string title { get; set; }
public string genre { get; set; }
}
有人提交了歌曲列表:
List<Song> songs = new List<Song> {
new Song { Id = 1, title = "Song 1" , genre="rock"},
new Song { Id = 2, title = "Song 2" , genre="disco"}};
我想使用我的存储过程更新数据库,该过程可以插入新歌曲或在歌曲已存在时更新它。我的存储过程有两个输出参数:
@success_added int = 0 and @success_updated int = 0
我的存储过程如下:
ALTER PROCEDURE [dbo].[UpdateSong]
@Id int = 0,
@title varchar(25) = NULL,
@genre varchar(25) = NULL,
@success_updated bit = 0 OUTPUT,
@success_added bit = 0 OUTPUT
AS
IF NOT EXISTS (SELECT Id FROM Songs WHERE Id = @Id)
BEGIN
INSERT INTO Songs
(
-- Id created by db
title,
genre
)
VALUES
(
@title,
@genre
)
SELECT @Success_Added = 1, @Success_Updated = 0
END
ELSE -- song already exists
BEGIN
UPDATE Songs
SET
title = @title,
@genre = @genre
WHERE Id = @Id
SELECT @Success_Added = 0, @Success_Updated = 1
END
RETURN
我知道这行得通:
dbConn.Open();
DynamicParameters p = new DynamicParameters();
foreach (var song in songs)
{
p.Add("@Id", song.Id);
p.Add("@title", song.title);
p.Add("@genre", song.genre);
p.Add("@success_updated", dbType: DbType.Boolean, direction: ParameterDirection.Output);
p.Add("@success_added", dbType: DbType.Boolean, direction: ParameterDirection.Output);
dbConn.Execute("Test_UpdateSong", p, commandType: CommandType.StoredProcedure);
Console.WriteLine("@success_added: " + p.Get<Boolean>("@success_added"));
Console.WriteLine("@success_updated: " + p.Get<Boolean>("@success_updated"));
}
dbConn.Close();
但这需要手动将每首歌曲 属性 转换为匿名类型 DynamicParameter。我宁愿简单地这样做:
dbConn.Open();
foreach (var song in songs)
{
var updateResult = dbConn.Query<dynamic>("Test_UpdateSong", song, commandType: CommandType.StoredProcedure);
}
dbConn.Close();
这也行得通。但是,现在如何获取我的输出参数?
正如我最初所说,我不想手动将每个 class 属性 转换为 Dapper 动态参数。这很关键,因为如果我创建一个泛型方法,我可能不知道哪个 class 被传递到该方法中,因此不知道要传递哪些属性以转换为动态参数。采纳@Metro Smurfs 的建议(如果其他方法都失败了,请阅读说明),我查看了 Dapper 测试 class 并找到了适合我的解决方案:
DynamicParameters p = new DynamicParameters(song);
通过将歌曲对象添加到 DynamicParameters 构造函数,创建了一个 DynamicParameters 模板,它会自动将所有属性转换为参数。然后我可以简单地添加我的两个输出参数并执行 sproc:
p.Add("@success_updated", dbType: DbType.Boolean, direction: ParameterDirection.Output);
p.Add("@success_added", dbType: DbType.Boolean, direction: ParameterDirection.Output);
dbConn.Execute("Test_UpdateSong", p, commandType: CommandType.StoredProcedure);
// get my output parameters...
var success_added = p.Get<bool>("@success_Added");
var success_added = p.Get<bool>("@success_Updated");
我准备好了!感谢@Nick 和@Metro Smurf 的建议!