将匿名类型的强制列表列表到动态对象列表

时间:2013-04-24 03:38:37

标签: c# .net-4.0 anonymous-types

为什么我不能将List<AnonymousObject>投放到List<dynamic>?我有以下代码:

var datasource = someList.Select(o => new { x = o.A, y = o.B });
dgvSomeGridView.DataSource = datasource.ToList();
dgvSomeGridView.DataBind();

然后我使用以下代码访问GridView.DataSource

var ds = ((List<dynamic>)dgvSomeGridView.DataSource);
....

但它在我将其转换为List<dynamic>的行上引发错误,它说:

  

无法将System.Collections.Generic.List'1[<>f__AnonymousType0'8[System.Int32,System.String]]类型的对象转换为System.Collections.Generic.List'1[System.Object]类型。

为什么我不能将匿名类型列表转换为dynamic,或者错误说明为object类型?我该如何解决这个问题?

我的代码在C#中,框架4.0,在VS2010 Pro中构建,平台是ASP.NET。

请提前帮助,谢谢。

2 个答案:

答案 0 :(得分:40)

由于List<T>是变体,而不是变体,因此您必须强制进入支持共变体的IEnumerable<dynamic>

var ds = ((IEnumerable<dynamic>)dgvSomeGridView.DataSource).ToList();

了解更多information about covariant

答案 1 :(得分:7)

首先,使用泛型进行转换不会那样。此演员表无效:

List<string> source = GetStrings();
List<object> source2 = (List<object>) source;

原因是List不是共变体。如果是,你可以source2.Add(source2);,当它应该只有字符串时,突然source1包含自己。

其次,Anonymous types只是编译器声明的类,具有readonly属性和值相等语义。如果您创建了一个具有readonly属性和值相等语义的类,则您的类将与匿名类型相同,除非您的类型具有开发人员确定的名称,而匿名类型具有编译器确定的名称。匿名类型并不特殊。

第三,dynamic变量是一种绕过编译器类型检查的方法。它们不会进行运行时类型检查。您可以使用c#casting语法将类型显式转换为动态...注意:这不是强制转换!您不能对运行时不存在的类型执行运行时转换。

  

但是,包含dynamic类型表达式的操作未由编译器解析或进行类型检查。编译器将有关操作的信息打包在一起,该信息稍后用于评估运行时的操作。作为过程的一部分,dynamic类型的变量被编译为object类型的变量。因此,类型dynamic仅在编译时存在,而不是在运行时存在。

static void convertToDynamic()
{
    dynamic d;
    int i = 20;
    d = (dynamic)i;
    Console.WriteLine(d);

    string s = "Example string.";
    d = (dynamic)s;
    Console.WriteLine(d);

    DateTime dt = DateTime.Today;
    d = (dynamic)dt;
    Console.WriteLine(d);

}
// Results: 
// 20 
// Example string. 
// 2/17/2009 9:12:00 AM

最后,如果您仍想要List<dynamic>,请执行以下操作:

var anonList = GetAnonList();
List<dynamic> dynamicList = anonList.Select(x => (dynamic)x).ToList();

但你可以轻松地做到这一点:

var anonList = GetAnonList();
List<object> objectList = anonList.Cast<object>().ToList();