繁体   English   中英

如何使用 Dapper 存储过程调用父子调用

[英]How to call parent child call using Dapper Stored Procedure

以下是对象结构。

public class User{
   public string Name{get;set;}
   public IList<Address> Addresss {get;set;}
   ...
}

public class Addresss{
   public string Street {get;set;}
   ...
}

使用 Dapper,如何编写它来检索用户和地址列表,以及使用存储过程调用。

试图调用, DbConnection.QueryAsync<User>("storedprocedure",param:null,commandType:CommandType.StoredProcedure)

存储过程查询为, Select u.*,a.* from user u join address a on u.Id = a.UserId

预期结果为用户列表,其中 User.Address // 应填充关联地址列表。

我假设你的存储过程是这样的

SELECT u.Id, u.Name, a.UserId, a.Street FROM Users u JOIN Addresses a on u.Id = a.UserId

在这种情况下,您可以调用存储过程并以这种方式设置地址列表的元素

Dictionary<int, User> users = new Dictionary<int, User();
var result = connection.Query<User, Address, User>(spName, ((u, a) =>
{
    if (!users.ContainsKey(u.Id))
    {
        users.Add(u.Id, u);
        u.Addresses = new List<Address>();
    }
    User k = users[b.Id];
    k.Addresses.Add(a);
    return u;
}, splitOn:"UserId", commandType:CommandType.StoredProcedure);

那么,这里发生了什么? Query 方法中的 lambda 表达式接收 Dapper 从返回的 SP 数据中提取的两个对象(一个用户和一个地址),并按照 splitOn 参数所述在 UserId 字段处进行拆分。
然后 lambda 会检查字典中是否存在具有该 ID 的用户,如果没有,则添加用户及其键,而不会忘记初始化地址列表。
在 if 之后,lambda 从字典中取回用户并添加地址实例,最后返回作为输入参数接收的 User 对象。 当 Dapper 完成枚举结果时,IEnumerable 将与地址数据一起返回。

如果您期望只有 1 个用户,那么另一种方法可能是使用 QueryMultiple,其中存储过程必须返回 2 个数据表,第一个包含用户信息,第二个包含用户地址。

using (SqlConnection conn = new SqlConnection(this.DbContext.CadenaConexion))
{
    using (var results = conn.QueryMultiple($"EXEC MyStoredProcedure @userId=@userId", new { userId= 123 }))
    {
        User user = results.ReadFirst<User>();
        user.Addresses = results.Read<Address>().ToList();
        
        return user;
    }
}

存储过程查询如:

SELECT * FROM dbo.User WHERE Id = @userid
SELECT * FROM dbo.Address WHERE UserId = @userid

PS:我没有运行代码,应该不会变化太大

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM