12

在 C# 中,我想弄清楚是否可以声明一个匿名类型,其中字段直到运行时才知道。

例如,如果我有一个键/值对列表,我可以根据该列表的内容声明一个匿名类型吗?我正在处理的具体情况是将参数传递给 Dapper,我不知道我将有多少参数。

List<Tuple<string, string>> paramList = new List<Tuple<string, string>>() {
    new Tuple<string, string>("key1", "value1"),
    new Tuple<string, string>("key2", "value2")
    ...
};

我想将此 List(或等效的 Map)转换为可以作为查询参数传递给 Dapper 的匿名类型。所以理想情况下,如果定义为匿名类型,上面的列表最终会看起来像这样:

new { key1=value1, key2=value2, ... }

我在 StackOverflow 上看到了几个关于在声明匿名类型后扩展匿名类型(“extendo objects”)的问题,或者在创建对象后在对象上声明任意字段,但我不需要这样做......我只需要预先动态声明类型一次。我的怀疑是,如果可能的话,这将需要一些花哨的反思。

我的理解是编译器在编译时为匿名类定义了一个类型,所以如果该类的字段直到运行时才可用,我可能会不走运。实际上,我的用例实际上可能与使用“扩展对象”来定义任意字段没有什么不同。

或者,如果有人知道将查询参数传递给 Dapper 的更好方法(而不是声明一个匿名类),我也很想听听。

谢谢!

更新

很抱歉延迟回到这个!这些答案都很棒,希望能给大家加分。我最终使用了 jbtule 的解决方案(由 Sam Saffron 编辑),将 IDynamicParameters 传递给 Dapper,所以我觉得我必须给他答案。其他答案也很好,回答了我提出的具体问题。我真的很感谢大家在这方面的时间!

4

3 回答 3

13

Dapper 的创造者非常清楚这个问题。INSERTUPDATE助手确实需要这种功能。

,Query和方法接受一个Execute参数。这可以是匿名类型、具体类型或实现的对象。QueryMultipledynamicIDynamicParameters

public interface IDynamicParameters
{
    void AddParameters(IDbCommand command, Identity identity);
}

这个接口非常方便,AddParameters在运行任何 SQL 之前调用。这不仅使您可以对发送到 SQL 的参数进行丰富的控制。它允许您连接特定于数据库的 DbParameters,因为您可以访问该命令(您可以将其转换为特定于数据库的命令)。这允许支持表值参数等。

Dapper 包含此接口的实现,可用于您的目的,称为DynamicParameters. 这允许您连接匿名参数包并添加特定值。

您可以使用AddDynamicParams方法来附加匿名类型。

var p = new DynamicParameters();
p.AddDynamicParams(new{a = "1"});
p.AddDynamicParams(new{b = "2", c = "3"});
p.Add("d", "4")
var r = cnn.Query("select @a a, @b b, @c c, @d d", p);
// r.a == 1, r.b == 2, r.c == 3, r.d == 4
于 2011-09-15T20:53:39.557 回答
11

在 C# 中,我想弄清楚是否可以声明一个匿名类型,其中字段直到运行时才知道。

匿名类型由编译器生成。您想知道编译器是否会为您生成编译器生成的类型,其中包含编译器不知道的字段类型。显然它不能这样做;正如您正确猜测的那样,您很不走运。

我在 StackOverflow 上看到了几个关于在声明匿名类型后扩展匿名类型的问题(“extendo 对象”)

我们通常称这些为“expando”对象。

如果您想做的是基于键值对字典创建一个 expando 对象,那么请使用 ExpandoObject 类来执行此操作。有关详细信息,请参阅此 MSDN 文章:

http://msdn.microsoft.com/en-us/magazine/ff796227.aspx

如果您想要在运行时生成一个真正的 .NET 类,您也可以这样做。正如您正确指出的那样,您需要一些花哨的反射才能做到这一点。您要做的是制作一个可收集的程序集(之所以这么说是因为与普通程序集不同,您在运行时生成它,并且垃圾收集器会在您完成后清理它。)

请参阅http://msdn.microsoft.com/en-us/library/dd554932.aspx,了解有关如何制作可收藏程序集并使用 TypeBuilder 向其中发出类型的详细信息。

于 2011-09-15T20:31:59.063 回答
7

您不能使用匿名类型。匿名类型由编译器生成,而不是在运行时生成。你当然可以使用dynamic

dynamic dynamicObj = new ExpandoObject();    
var objAsDict = (IDictionary<String, Object>)dynamicObj;

foreach(var item in paramList)
{
    objAsDict.Add(item.Item1, item.Item2);
}

然后,您可以将 dynamicObj 用作常规对象:

Console.WriteLine(dynamicObj.key1); // would output "value1"
于 2011-09-15T20:30:18.193 回答