匿名类的通用列表

DHo*_*out 398 .net c# generics anonymous-types

在C#3.0中,您可以使用以下语法创建匿名类

var o = new { Id = 1, Name = "Foo" };
Run Code Online (Sandbox Code Playgroud)

有没有办法将这些匿名类添加到通用列表?

例:

var o = new { Id = 1, Name = "Foo" };
var o1 = new { Id = 2, Name = "Bar" };

List<var> list = new List<var>();
list.Add(o);
list.Add(o1);
Run Code Online (Sandbox Code Playgroud)

另一个例子:

List<var> list = new List<var>();

while (....)
{
    ....
    list.Add(new {Id = x, Name = y});
    ....
}
Run Code Online (Sandbox Code Playgroud)

Jon*_*eet 411

你可以这样做:

var list = new[] { o, o1 }.ToList();
Run Code Online (Sandbox Code Playgroud)

有很多方法可以为这只猫设置皮肤,但基本上它们都会在某处使用类型推断 - 这意味着你必须调用泛型方法(可能作为扩展方法).另一个例子可能是:

public static List<T> CreateList<T>(params T[] elements)
{
     return new List<T>(elements);
}

var list = CreateList(o, o1);
Run Code Online (Sandbox Code Playgroud)

你明白了:)

  • @DHornpout:这会给出一个数组,而不是List <T>. (32认同)
  • @DHornpout:你有"使用System.Linq;" 在你的文件的顶部?ToList是一个LINQ运算符. (23认同)
  • (当然,您还需要对System.Core的引用.) (8认同)
  • 得到它..需要包括"使用System.Linq".谢谢. (5认同)
  • 在Visual Studio中看起来似乎不一致,智能感知在发现缺少包含引用扩展方法的程序集(与引用类型相同)方面没有多大帮助. (2认同)
  • 这个男人到处都是,今天搜了8个问题,7回答他. (2认同)

小智 101

这是答案.

string result = String.Empty;

var list = new[]
{ 
    new { Number = 10, Name = "Smith" },
    new { Number = 10, Name = "John" } 
}.ToList();

foreach (var item in list)
{
    result += String.Format("Name={0}, Number={1}\n", item.Name, item.Number);
}

MessageBox.Show(result);
Run Code Online (Sandbox Code Playgroud)

  • Dutt,你的代码应该在没有.ToList()的情况下工作. (12认同)
  • 好吧,现在我们需要一个用select语句替换new {}行的例子.var list = sourceList.Select(o => new {o.ModelId,o.PartNumber,o.Quantity}).ToList(); (2认同)

小智 55

有很多方法可以做到这一点,但是这里的一些响应是创建一个包含垃圾元素的列表,这需要您清除列表.

如果要查找泛型类型的空列表,请使用"针对元组列表选择"来创建空列表.不会实例化任何元素.

这是创建空列表的单线程:

 var emptyList = new List<Tuple<int, string>>()
          .Select(t => new { Id = t.Item1, Name = t.Item2 }).ToList();
Run Code Online (Sandbox Code Playgroud)

然后您可以使用您的泛型类型添加到它:

 emptyList.Add(new { Id = 1, Name = "foo" });
 emptyList.Add(new { Id = 2, Name = "bar" });
Run Code Online (Sandbox Code Playgroud)

作为替代方案,您可以执行类似下面的操作来创建空列表(但是,我更喜欢第一个示例,因为您也可以将它用于填充的元组集合):

 var emptyList = new List<object>()
          .Select(t => new { Id = default(int), Name = default(string) }).ToList();   
Run Code Online (Sandbox Code Playgroud)

  • 谢谢你的想法。一个建议,如果您使用`Enumerable.Empty&lt;object&gt;().Select(o=&gt;definition).ToList()`,则可以避免分配虚拟列表 (2认同)

Jef*_*ser 44

不完全是,但你可以说List<object>,事情会奏效.但是,list[0].Id不会起作用.

这将在运行时在C#4.0中通过拥有一个List<dynamic>,即你将无法获得IntelliSense.

  • 这是我担心人们会用动态做的事情. (29认同)
  • 我没有说这是一个很棒的想法,但它有可能:-)例如,可能需要从Ruby存储对象. (2认同)
  • 但是在这些情况下,源类型毕竟是动态的,对于匿名类型使用List <dynamic>是没有意义的. (2认同)

eri*_*len 23

我猜

List<T> CreateEmptyGenericList<T>(T example) {
    return new List<T>();
}

void something() {
    var o = new { Id = 1, Name = "foo" };
    var emptyListOfAnonymousType = CreateEmptyGenericList(o);
}
Run Code Online (Sandbox Code Playgroud)

将工作.

你也可以考虑这样写:

void something() {
    var String = string.Emtpy;
    var Integer = int.MinValue;
    var emptyListOfAnonymousType = CreateEmptyGenericList(new { Id = Integer, Name = String });
}
Run Code Online (Sandbox Code Playgroud)


Ros*_*tov 20

我通常使用以下内容; 主要是因为你然后用一个空的列表"开始".

var list = Enumerable.Range(0, 0).Select(e => new { ID = 1, Name = ""}).ToList();
list.Add(new {ID = 753159, Name = "Lamont Cranston"} );
//etc.
Run Code Online (Sandbox Code Playgroud)

最近,我一直这样写它:

var list = Enumerable.Repeat(new { ID = 1, Name = "" }, 0).ToList();
list.Add(new {ID = 753159, Name = "Lamont Cranston"} );
Run Code Online (Sandbox Code Playgroud)

使用repeat方法还可以执行以下操作:

var myObj = new { ID = 1, Name = "John" };
var list = Enumerable.Repeat(myObj, 1).ToList();
list.Add(new { ID = 2, Name = "Liana" });
Run Code Online (Sandbox Code Playgroud)

..which为您提供已添加的第一个项目的初始列表.

  • 您不需要以空列表开头 - 您可以执行Range(0,1)并使select语句中的第一个对象成为第一个对象. (2认同)
  • 您不需要从一个空列表开始——如果您知道第一项是什么(如示例中所示),那么您就在评论中。很多时候,虽然我使用它来解析中间文件/数据源,并且在 LINQ 投影场景中使用它之前不访问第一个真正的项目(因此不需要考虑跳过第一条记录)。 (2认同)

Mal*_*eBR 19

您可以在代码中执行此操作.

var list = new[] { new { Id = 1, Name = "Foo" } }.ToList();
list.Add(new { Id = 2, Name = "Bar" });
Run Code Online (Sandbox Code Playgroud)


Bas*_*sem 17

如果您使用的是 C# 7 或更高版本,则可以使用元组类型而不是匿名类型。

var myList = new List<(int IntProp, string StrProp)>();
myList.Add((IntProp: 123, StrProp: "XYZ"));
Run Code Online (Sandbox Code Playgroud)


MEC*_*MEC 10

我在几个答案上检查了IL.此代码有效地提供了一个空列表:

    using System.Linq;
    …
    var list = new[]{new{Id = default(int), Name = default(string)}}.Skip(1).ToList();
Run Code Online (Sandbox Code Playgroud)


Dam*_*nka 9

在最新的4.0版本中,可以使用如下的动态

var list = new List<dynamic>();
        list.Add(new {
            Name = "Damith"
    });
        foreach(var item in list){
            Console.WriteLine(item.Name);
        }
    }
Run Code Online (Sandbox Code Playgroud)


use*_*r_v 8

这是我的尝试.

List<object> list = new List<object> { new { Id = 10, Name = "Testing1" }, new {Id =2, Name ="Testing2" }}; 
Run Code Online (Sandbox Code Playgroud)

当我为自定义类型制作匿名列表时,我想出了类似的东西.


Tom*_*Dee 8

我很惊讶没有人建议使用集合初始值设定项。这种方式只能在创建列表时添加对象,因此名称,但它似乎是最好的方式。无需创建数组然后将其转换为列表。

var list = new List<dynamic>() 
{ 
    new { Id = 1, Name = "Foo" }, 
    new { Id = 2, Name = "Bar" } 
};
Run Code Online (Sandbox Code Playgroud)

您始终可以使用object而不是dynamic尝试以真正的通用方式保留它,然后dynamic更有意义。


小智 7

下面是另一种创建匿名类型列表的方法,它允许您从空列表开始,但仍然可以访问IntelliSense.

var items = "".Select( t => new {Id = 1, Name = "foo"} ).ToList();
Run Code Online (Sandbox Code Playgroud)

如果你想保留第一个项目,只需在字符串中放一个字母.

var items = "1".Select( t => new {Id = 1, Name = "foo"} ).ToList();
Run Code Online (Sandbox Code Playgroud)


Cod*_*ack 7

您可以创建一个动态列表。

List<dynamic> anons=new List<dynamic>();
foreach (Model model in models)
{
   var anon= new
   {
      Id = model.Id,
      Name=model.Name
   };
   anons.Add(anon);
}
Run Code Online (Sandbox Code Playgroud)

“动态”通过添加的第一个值进行初始化。


小智 6

你可以这样做:

var o = new { Id = 1, Name = "Foo" };
var o1 = new { Id = 2, Name = "Bar" };

var array = new[] { o, o1 };
var list = array.ToList();

list.Add(new { Id = 3, Name = "Yeah" });
Run Code Online (Sandbox Code Playgroud)

这对我来说似乎有点“hacky”,但它有效 - 如果你真的需要一个列表并且不能只使用匿名数组。


小智 6

而不是这个:

var o = new { Id = 1, Name = "Foo" }; 
var o1 = new { Id = 2, Name = "Bar" }; 

List <var> list = new List<var>(); 
list.Add(o); 
list.Add(o1);
Run Code Online (Sandbox Code Playgroud)

你可以这样做:

var o = new { Id = 1, Name = "Foo" }; 
var o1 = new { Id = 2, Name = "Bar" }; 

List<object> list = new List<object>(); 
list.Add(o); 
list.Add(o1);
Run Code Online (Sandbox Code Playgroud)

但是,如果您尝试在另一个范围内执行此类操作,则会遇到编译时错误,尽管它在运行时有效:

private List<object> GetList()
{ 
    List<object> list = new List<object>();
    var o = new { Id = 1, Name = "Foo" }; 
    var o1 = new { Id = 2, Name = "Bar" }; 
    list.Add(o); 
    list.Add(o1);
    return list;
}

private void WriteList()
{
    foreach (var item in GetList()) 
    { 
        Console.WriteLine("Name={0}{1}", item.Name, Environment.NewLine); 
    }
}
Run Code Online (Sandbox Code Playgroud)

问题是只有Object的成员在运行时可用,尽管intellisense将显示属性idname.

在.net 4.0中,解决方案是在上面的代码中使用关键字dynamic istead of object.

另一种解决方案是使用反射来获取属性

using System;
using System.Collections.Generic;
using System.Reflection;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            Program p = new Program();
            var anonymous = p.GetList(new[]{
                new { Id = 1, Name = "Foo" },       
                new { Id = 2, Name = "Bar" }
            });

            p.WriteList(anonymous);
        }

        private List<T> GetList<T>(params T[] elements)
        {
            var a = TypeGenerator(elements);
            return a;
        }

        public static List<T> TypeGenerator<T>(T[] at)
        {
            return new List<T>(at);
        }

        private void WriteList<T>(List<T> elements)
        {
            PropertyInfo[] pi = typeof(T).GetProperties();
            foreach (var el in elements)
            {
                foreach (var p in pi)
                {
                    Console.WriteLine("{0}", p.GetValue(el, null));
                }
            }
            Console.ReadLine();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)


小智 5

var list = new[]{
new{
FirstField = default(string),
SecondField = default(int),
ThirdField = default(double)
}
}.ToList();
list.RemoveAt(0);
Run Code Online (Sandbox Code Playgroud)


小智 5

这是一个老问题,但是我想我应该填写C#6答案。我经常必须设置易于在代码中以元组列表形式输入的测试数据。有了几个扩展功能,就可以拥有这种精巧紧凑的格式,而无需在每个条目上重复名称。

var people= new List<Tuple<int, int, string>>() {
    {1, 11, "Adam"},
    {2, 22, "Bill"},
    {3, 33, "Carol"}
}.Select(t => new { Id = t.Item1, Age = t.Item2, Name = t.Item3 });
Run Code Online (Sandbox Code Playgroud)

这提供了IEnumerable-如果您想要可以添加到的列表,则只需添加ToList()。

魔术来自用于元组的自定义扩展Add方法,如/sf/answers/1921907571/所述

public static class TupleListExtensions    {
    public static void Add<T1, T2>(this IList<Tuple<T1, T2>> list,
            T1 item1, T2 item2)       {
        list.Add(Tuple.Create(item1, item2));
    }

    public static void Add<T1, T2, T3>(this IList<Tuple<T1, T2, T3>> list,
            T1 item1, T2 item2, T3 item3) {
        list.Add(Tuple.Create(item1, item2, item3));
    }

// and so on...
Run Code Online (Sandbox Code Playgroud)

}

我唯一不喜欢的是类型与名称分开,但是如果您真的不想创建一个新类,那么这种方法仍然可以让您拥有可读的数据。