百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

C#.NET 匿名对象详解_c# 匿名委托

wptr33 2025-09-01 15:47 7 浏览

简介

匿名对象(Anonymous Types)是一种在编译时由编译器自动生成、但在源码中没有显式命名的引用类型,用来快速封装一组只读属性。它们最常见的场景是在 LINQ 查询中临时投影数据,但也可用于任何需要临时封装数据的地方。

基本语法

// 最简单的匿名对象,两个属性:Name(string)、Age(int)
var person = new { Name = "Alice", Age = 30 };

// 访问属性
Console.WriteLine(person.Name);  // 输出 "Alice"
Console.WriteLine(person.Age);   // 输出 30
  • 关键字:使用 new { … },在大括号里列出“属性名 = 值”对。
  • 类型:编译器在后台生成一个内部 sealed 类,属性带 get 访问器且只读。
  • 推断:使用 var 才能声明匿名类型变量;不能写成 SomeType person = ...。

属性特性

  • 只读:匿名类型的属性只带 get,没有 set
  • 自动 Equals/GetHashCode:编译器重写了这两个方法,使得如果两个匿名实例的属性名称、顺序、类型和值都相同,则它们相互 Equals 并且哈希值相同。
  • ToString():被重写以返回类似 { Name = Alice, Age = 30 } 的字符串。
var a = new { X = 1, Y = 2 };
var b = new { X = 1, Y = 2 };
Console.WriteLine(a.Equals(b));      // True
Console.WriteLine(a.GetHashCode());  // 与 b 相同
Console.WriteLine(a.ToString());     // "{ X = 1, Y = 2 }"

底层原理

编译器会自动生成类似如下的类:

[CompilerGenerated]
internal sealed class <>f__AnonymousType0<<Name>j__TPar, <Age>j__TPar>
{
    public string Name { get; }
    public int Age { get; }
    
    public <>f__AnonymousType0(string name, int age)
    {
        this.Name = name;
        this.Age = age;
    }
    
    // 自动生成的 Equals, GetHashCode, ToString 方法
}

与命名类型的对比

特性

匿名对象

命名类/结构体

定义

编译器自动生成,无源码名称

必须手动定义

属性可写性

只读

可读可写或只读

Equals/GetHashCode

自动基于所有属性实现

需手动重写

ToString

自动按属性输出

默认输出类型全名,或手动重写

可复用性

仅在本方法/本表达式作用域使用

全局可见、可复用

用途

临时封装、LINQ 投影

长期持有、跨层传递

在 LINQ 中的典型用法

匿名对象最常见于 LINQselect 投影,将原始类型映射到只关心的字段集合:

var people = new[]
{
    new { Name = "Alice", Age = 30, Country = "USA" },
    new { Name = "Bob",   Age = 25, Country = "UK" },
    new { Name = "Carol", Age = 28, Country = "USA" },
};

var query = people
    .Where(p => p.Country == "USA")
    .Select(p => new { p.Name, p.Age });  // 只投影 Name 和 Age

foreach (var item in query)
    Console.WriteLine(#34;{item.Name} is {item.Age} years old");
  • new { p.Name, p.Age } 是属性名与源属性同名的简写,相当于 new { Name = p.Name, Age = p.Age }

类型推断与作用域

  • 匿名类型在同一程序集中同一属性名/顺序/类型出现时会被视为“同一类型”。
  • 如果在不同方法或不同项目(不同程序集)里定义了形状相同的匿名对象,它们并不是相同的类型,不能互相赋值。
// 方法 A
var p1 = new { X = 1, Y = 2 };
// 方法 B
var p2 = new { X = 3, Y = 4 };
// 即使属性完全一样,p1、p2 在不同方法范围内,也有相同编译时类型
// 但在不同项目或不同编译单元里,类型不同,不能跨程序集传递

限制与注意事项

  • 只能有属性:匿名类型不能定义方法、字段或事件,只能有公有只读属性。
  • 不可作为方法返回类型显式声明:只能用 var 接受,或返回 object/接口/动态,但这样会丢失静态类型安全。
  • 不可用于公开 API:不宜在公共方法签名中使用匿名类型,因外部无法引用其实际类型。
  • 不可修改:所有属性只读;如果需要改值,需创建新实例。

与 dynamic、元组对比

特性

匿名对象

dynamic

值元组 (… )

静态类型

静态类型(有编译期检查)

动态类型(仅运行期检查)

静态类型

属性命名

自定义命名

任意命名

Item1/Item2 或具名元组字段

只读/可写

只读

可读可写

可读可写

Equals

基于属性自动实现

调用动态对象的 Equals

元组值相等比较

用途

临时封装、LINQ 投影

运行期灵活、COM/反射场景

临时打包多值返回

使用限制

// 1. 不能作为方法参数或返回值
public void Process(/* 错误:var data */) { } 

// 2. 不能在类中定义字段
class MyClass {
    // private var _data; // 错误
}

// 3. 属性只读
var item = new { Value = 10 };
// item.Value = 20; // 编译错误

// 4. 不能添加方法
var calculator = new { 
    // int Add(int a, int b) => a + b; // 不允许
};

高级用法

进阶:在方法间传递匿名对象

虽然不能在方法签名里写明匿名类型,但可以利用 泛型 与 类型推断:

T Echo<T>(T obj) => obj;

// 调用时:
var anon = new { Name = "X", Age = 99 };
var same = Echo(anon);   // T 推断为该匿名类型
Console.WriteLine(same.Name);  // 依然可访问

嵌套匿名对象

var company = new {
    Name = "TechCorp",
    Address = new {
        Street = "123 Main St",
        City = "Seattle"
    },
    Employees = new[] {
        new { Name = "Alice", Position = "Developer" },
        new { Name = "Bob", Position = "Manager" }
    }
};

Console.WriteLine(company.Address.City); // Seattle

动态类型转换

// 匿名对象转动态类型
dynamic dynamicPerson = new {
    FirstName = "John",
    LastName = "Doe"
};

Console.WriteLine(dynamicPerson.FirstName); // John

// 动态类型转匿名对象(需类型匹配)
var anonymousPerson = (object)dynamicPerson as dynamic;

JSON 序列化

using System.Text.Json;

var data = new {
    Timestamp = DateTime.UtcNow,
    Event = "UserLogin",
    Details = new {
        UserId = 12345,
        IP = "192.168.1.1"
    }
};

string json = JsonSerializer.Serialize(data);
// 输出: 
// {"Timestamp":"2023-07-15T12:30:45Z","Event":"UserLogin","Details":{"UserId":12345,"IP":"192.168.1.1"}}

Web API 临时响应

[HttpGet("user/{id}")]
public IActionResult GetUserSummary(int id)
{
    var user = _dbContext.Users.Find(id);
    if (user == null) return NotFound();
    
    // 使用匿名对象构建响应
    return Ok(new {
        user.Id,
        user.Username,
        JoinDate = user.CreatedAt.ToString("yyyy-MM-dd"),
        PostCount = _dbContext.Posts.Count(p => p.UserId == id)
    });
}

数据转换管道

public IEnumerable<dynamic> TransformData(IEnumerable<DataRecord> records)
{
    return records
        .Where(r => r.IsValid)
        .Select(r => new {
            Key = r.Id.ToString("D5"),
            Value = r.Amount * r.ConversionRate,
            Category = GetCategory(r.Type)
        })
        .GroupBy(x => x.Category)
        .Select(g => new {
            Category = g.Key,
            Total = g.Sum(x => x.Value),
            Items = g.ToList()
        });
}

单元测试数据准备

[Test]
public void CalculateTotal_ShouldReturnCorrectSum()
{
    // 准备测试数据
    var items = new[] {
        new { Name = "Item1", Price = 10.99m, Quantity = 2 },
        new { Name = "Item2", Price = 5.49m, Quantity = 3 },
        new { Name = "Item3", Price = 15.00m, Quantity = 1 }
    };
    
    // 执行测试
    decimal total = Calculator.CalculateTotal(items);
    
    // 验证结果
    Assert.AreEqual(10.99m*2 + 5.49m*3 + 15.00m, total);
}

结合动态类型

dynamic dynObj = new { Message = "Hello", Code = 200 };
Console.WriteLine(dynObj.Message); // 运行时解析

// 匿名类型转动态 ExpandoObject
dynamic expando = new ExpandoObject();
var anon = new { A = 1, B = 2 };
foreach (var prop in anon.GetType().GetProperties())
{
    ((IDictionary<string, object>)expando)[prop.Name] = prop.GetValue(anon);
}

反射操作

var anon = new { Title = "C# Guide", Pages = 300 };

// 读取属性
PropertyInfo prop = anon.GetType().GetProperty("Title");
string value = (string)prop.GetValue(anon); // "C# Guide"

// 动态创建匿名类型
Type anonType = RuntimeHelpers.GetAnonymousType(
    new { Name = default(string), Age = default(int) }.GetType()
);
object newObj = Activator.CreateInstance(anonType, "Tom", 25);

动态查询构建器

public static IQueryable<object> BuildDynamicQuery(
    IQueryable<User> source, 
    params string[] fields)
{
    var parameter = Expression.Parameter(typeof(User), "u");
    var bindings = fields.Select(field => 
        Expression.Bind(
            typeof(User).GetProperty(field),
            Expression.Property(parameter, field)
        )
    );
    
    var body = Expression.MemberInit(
        Expression.New(typeof(User)), 
        bindings
    );
    
    var selector = Expression.Lambda(body, parameter);
    return source.Select((dynamic)selector);
}

// 使用:仅查询指定字段
var result = BuildDynamicQuery(db.Users, "Id", "Username").ToList();

相关推荐

栋察宇宙(二十一):Python 文件操作全解析

分享乐趣,传播快乐,增长见识,留下美好。亲爱的您,这里是LearingYard学苑!...

python中12个文件处理高效技巧,不允许你还不知道

在Python中高效处理文件是日常开发中的核心技能,尤其是处理大文件或需要高性能的场景。以下是经过实战验证的高效文件处理技巧,涵盖多种常见场景:一、基础高效操作...

Python内置模块bz2: 对 bzip2压缩算法的支持详解

目录简介知识讲解2.1bzip2压缩算法原理2.2bz2模块概述...

Python文件及目录处理方法_python目录下所有文件名

Python可以用于处理文本文件和二进制文件,比如创建文件、读写文件等操作。本文介绍Python处理目录以及文件的相关方法。...

The West mustn&#39;t write China out of WWII any longer

ByWarwickPowellLead:Foreightdecades,theWesthasrewrittenWorldWarIIasanAmericanandEuro...

Python 的网络与互联网访问模块及应用实例(一)

Python提供了丰富的内置模块和第三方库来处理网络与互联网访问,使得从简单的HTTP请求到复杂的网络通信都变得相对简单。以下是常用的网络模块及其应用实例。...

高效办公:Python处理excel文件,摆脱无效办公

一、Python处理excel文件1.两个头文件importxlrdimportxlwt...

Python进阶:文件读写操作详解_python对文件的读写操作方法有哪些

道友今天开始进阶练习,来吧文件读写是Python编程中非常重要的技能,掌握这些操作可以帮助你处理各种数据存储和交换任务。下面我将详细介绍Python中的文件读写操作。一、基本文件操作...

[827]ScalersTalk成长会Python小组第11周学习笔记

Scalers点评:在2015年,ScalersTalk成长会完成Python小组完成了《Python核心编程》第1轮的学习。到2016年,我们开始第二轮的学习,并且将重点放在章节的习题上。Pytho...

ScalersTalk 成长会 Python 小组第 9 周学习笔记

Scalers点评:在2015年,ScalersTalk成长会完成Python小组完成了《Python核心编程》第1轮的学习。到2016年,我们开始第二轮的学习,并且将重点放...

简析python 文件操作_python对文件的操作方法

一、打开并读文件1、file=open('打开文件的路径','打开文件的权限')#打开文件并赋值给file#默认权限为r及读权限str=read(num)读文件并放到字符串变量中,其中num表...

Python 中 必须掌握的 20 个核心函数——open()函数

open()是Python中用于文件操作的核心函数,它提供了读写文件的能力,是处理文件输入输出的基础。一、open()的基本用法1.1方法签名...

python常用的自动化脚本汇总_python 自动脚本

以下是python常用的自动化脚本,包括数据、网络、文件、性能等操作。具体内容如下:数据处理工具网络检测工具系统任务自动化工具测试自动化工具文件管理自动化工具性能监控工具日志分析工具邮件...

Python自动化办公应用学习笔记37—文件读写方法1

一、文件读写方法1.读取内容:read(size):读取指定大小的数据,如果不指定size,则读取整个文件。...

大叔转行SAP:好好学习,好好工作,做一个幸福的SAP人

我是一个崇尚努力的人,坚定认为努力可以改变命运和现状,同时也对自己和未来抱有非常高的期待。随着期待的落空,更对现状滋生不满,结果陷入迷茫。开始比较,发现周围人一个个都比你有钱,而你的事业,永远看不到明...