Jua*_*gal 6 c# orm ms-access dapper
我正在尝试执行一个简单的查询,结果数据几乎都是null.
我有这个表结构
表注册器
ID | Autonumeric
TareaM_Id | Numeric
Fecha | Date/Time
Run Code Online (Sandbox Code Playgroud)
和 Macro_tareas 表
ID | Autonumeric
Nombre | Short Text
Run Code Online (Sandbox Code Playgroud)
我在 C# 中映射了类,如下所示:
[Table("Registros")]
public class Registro
{
[Column("ID")]
public virtual int ID { get; set; }
[Column("Fecha")]
public virtual DateTime Fecha { get; set; }
[Column("TareaM_Id")]
public virtual int TareaM_Id { get; set; }
public virtual MacroTarea MacroT { get; set; }
}
[Table("Macro_tarea")]
public class MacroTarea
{
[Column("ID")]
public virtual int ID { get; set; }
[Column("Nombre")]
public virtual string Nombre{ get; set; }
public virtual ICollection<Registro> Registros { get; set; }
}
Run Code Online (Sandbox Code Playgroud)
这是我正在尝试使用的查询
string sql = @"SELECT reg.ID, mac.ID
FROM Registros as reg INNER JOIN Macro_tarea as mac on reg.TareaM_Id = mac.ID
WHERE Fecha = @Fecha";
using (IDbConnection db = new OleDbConnection(ConnectionString))
{
var result = db.Query<Registro,MacroTarea, Registro>(sql,
(reg,mac) =>
{
reg.MacroTarea = mac;
return reg;
}
,new { @Fecha = new DateTime(2019, 1, 4).Date }
, splitOn: "mac.ID")
.AsList();
}
Run Code Online (Sandbox Code Playgroud)
我试图只检索 id,但两个 id 都变为空,为什么会发生这种情况?
问题是,如果我将Registros.Fecha和添加Macro_tarea.Nombre到查询中,它就会正确获取值。但 id 一直为空。
显然这个问题只发生在 ids 上。我怀疑这个问题是由于重复的列名造成的。
我正在与 Microsoft Access 合作,这只是重要的事情。
我的问题与可能的重复问题不同,因为我定义了应该映射的类。
由于代码无法处理数据而重命名数据库列并不是一个好主意。在关注点分离的世界中,您的数据库为什么要关心?将 ID 列命名为“Id”有充分的数据库理由,您甚至可能没有更改它们的选项。
Dapper 映射还有另一个问题,即重命名列无法解决;重复类型。如果您尝试映射到类的多个实例,Dapper 会感到困惑,并且重命名列将不起作用,因为您将重命名两个实例。
这是我想出的解决方案。它与许多使用字典的示例类似,除了:
在此示例中,有一个包含许多批次的拍卖。每批可能有 1 件或多件物品。物品可能是物品包。物品来自有限的目录,我们喜欢关系数据,因此物品表包含每个物品的详细信息,如颜色、尺寸等。这里我们只获得单个批次,但获得拍卖与其他级别相同在拍卖的顶部。
参数 1 - 一次性获取所有内容的 SQL
参数 2 - 我们将返回的每个对象的类型数组。因此,最好命令 SELECT 将字段分组到类中
参数 3 - 使用 SQL 结果调用我们即将编写的方法
参数 4 - SQL 的标准参数数组。SQL 注入很糟糕,好吗?
public async Task<List<Lot>> GetAll(int auctionId)
{
using (var connection = new SqlConnection(_appSettings.ConnectionString))
{
await connection.OpenAsync();
var result = new List<Lot>();
await connection.QueryAsync($@"
SELECT [Lot].*,
[Item].[Id],
[Item].[LotId],
[Item].[Notes],
itemDetails.[Id],
itemDetails.[ThingId],
itemDetails.[Colour],
itemDetails.[Size],
[SubItem].[Id],
[SubItem].[ItemId],
[SubItem].[Notes],
subItemDetails.[Id],
subItemDetails.[ThinId],
subItemDetails.[Colour],
subItemDetails.[Size]
FROM [Lot]
INNER JOIN [Item] ON [Item].[LotId] = [Lot].[Id]
LEFT JOIN [Thing] AS itemDetails ON itemDetails.[Id] = [Item].[ThingId]
LEFT JOIN [SubItem] ON [SubItem].[ItemId] = [Item].[Id]
LEFT JOIN [Thing] AS subItemDetails ON subItemDetails.[Id] = [SubItem].[ThingId]
WHERE [AuctionId] = @{nameof(auctionId)}
ORDER BY [Lot].[Id], [Item].[Id], [Expansion].[Id];",
new Type[] {
typeof(Lot),
typeof(Item),
typeof(Thing),
typeof(Expansion),
typeof(Thing)
},
MapResult(result),
new
{
AuctionId = auctionId
}
);
return result.ToList();
}
}
private Func<object[], Lot> MapResult(List<Lot> result)
{
return (obj) =>
{
Lot lot = (Lot)obj[0];
Item item = (Item)obj[1];
Thing itemDetails = (Thing)obj[2];
SubItem subItem = (SubItem)obj[3];
Thing subItemDetails = (Thing)obj[4];
if (lot != null)
{
if (result.Any(a => a.Id == lot.Id))
{
lot = result.First(a => a.Id == lot.Id);
}
else
{
result.Add(lot);
}
}
if (item != null)
{
if (lot.Items.Any(i => i.Id == item.Id))
{
item = lot.Items.First(i => i.Id == item.Id);
}
else
{
lot.Items.Add(item.FromThing(itemDetails));
}
}
if (expansion != null)
{
if (item.SubItems.Any(e => e.Id == subItem.Id) == false)
{
item.SubItems.Add(subItem.FromThing(subItemDetails));
}
}
return null;
};
}
Run Code Online (Sandbox Code Playgroud)
MapResult 是代码的核心。它返回一个具有两种类型的 Func,即我们上面定义的 Type 数组和返回 Type,并采用顶级对象的 List。然后,我将对象数组中的每个项目映射到另一个实际类型。这使得代码更易于阅读,并且可以毫无问题地访问对象的属性和方法。
然后是逐步向下层次结构的情况,在每一步检查是否已经存在具有匹配 id 的迭代器,如果存在,则将迭代器交换为对其的引用。这意味着以下代码将添加到现有项目中。
在特定情况下,我还添加了 FromThing 函数,以便更轻松地组合对象属性。