C#.NET 匿名对象详解_c# 匿名委托
wptr33 2025-09-01 15:47 19 浏览
简介
匿名对象(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 中的典型用法
匿名对象最常见于 LINQ 的 select 投影,将原始类型映射到只关心的字段集合:
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();相关推荐
- oracle数据导入导出_oracle数据导入导出工具
-
关于oracle的数据导入导出,这个功能的使用场景,一般是换服务环境,把原先的oracle数据导入到另外一台oracle数据库,或者导出备份使用。只不过oracle的导入导出命令不好记忆,稍稍有点复杂...
- 继续学习Python中的while true/break语句
-
上次讲到if语句的用法,大家在微信公众号问了小编很多问题,那么小编在这几种解决一下,1.else和elif是子模块,不能单独使用2.一个if语句中可以包括很多个elif语句,但结尾只能有一个...
- python continue和break的区别_python中break语句和continue语句的区别
-
python中循环语句经常会使用continue和break,那么这2者的区别是?continue是跳出本次循环,进行下一次循环;break是跳出整个循环;例如:...
- 简单学Python——关键字6——break和continue
-
Python退出循环,有break语句和continue语句两种实现方式。break语句和continue语句的区别:break语句作用是终止循环。continue语句作用是跳出本轮循环,继续下一次循...
- 2-1,0基础学Python之 break退出循环、 continue继续循环 多重循
-
用for循环或者while循环时,如果要在循环体内直接退出循环,可以使用break语句。比如计算1至100的整数和,我们用while来实现:sum=0x=1whileTrue...
- Python 中 break 和 continue 傻傻分不清
-
大家好啊,我是大田。...
- python中的流程控制语句:continue、break 和 return使用方法
-
Python中,continue、break和return是控制流程的关键语句,用于在循环或函数中提前退出或跳过某些操作。它们的用途和区别如下:1.continue(跳过当前循环的剩余部分,进...
- L017:continue和break - 教程文案
-
continue和break在Python中,continue和break是用于控制循环(如for和while)执行流程的关键字,它们的作用如下:1.continue:跳过当前迭代,...
- 作为前端开发者,你都经历过怎样的面试?
-
已经裸辞1个月了,最近开始投简历找工作,遇到各种各样的面试,今天分享一下。其实在职的时候也做过面试官,面试官时,感觉自己问的问题很难区分候选人的能力,最好的办法就是看看候选人的github上的代码仓库...
- 面试被问 const 是否不可变?这样回答才显功底
-
作为前端开发者,我在学习ES6特性时,总被const的"善变"搞得一头雾水——为什么用const声明的数组还能push元素?为什么基本类型赋值就会报错?直到翻遍MDN文档、对着内存图反...
- 2023金九银十必看前端面试题!2w字精品!
-
导文2023金九银十必看前端面试题!金九银十黄金期来了想要跳槽的小伙伴快来看啊CSS1.请解释CSS的盒模型是什么,并描述其组成部分。...
- 前端面试总结_前端面试题整理
-
记得当时大二的时候,看到实验室的学长学姐忙于各种春招,有些收获了大厂offer,有些还在苦苦面试,其实那时候的心里还蛮忐忑的,不知道自己大三的时候会是什么样的一个水平,所以从19年的寒假放完,大二下学...
- 由浅入深,66条JavaScript面试知识点(七)
-
作者:JakeZhang转发链接:https://juejin.im/post/5ef8377f6fb9a07e693a6061目录...
- 2024前端面试真题之—VUE篇_前端面试题vue2020及答案
-
添加图片注释,不超过140字(可选)...
- 今年最常见的前端面试题,你会做几道?
-
在面试或招聘前端开发人员时,期望、现实和需求之间总是存在着巨大差距。面试其实是一个交流想法的地方,挑战人们的思考方式,并客观地分析给定的问题。可以通过面试了解人们如何做出决策,了解一个人对技术和解决问...
- 一周热门
- 最近发表
-
- oracle数据导入导出_oracle数据导入导出工具
- 继续学习Python中的while true/break语句
- python continue和break的区别_python中break语句和continue语句的区别
- 简单学Python——关键字6——break和continue
- 2-1,0基础学Python之 break退出循环、 continue继续循环 多重循
- Python 中 break 和 continue 傻傻分不清
- python中的流程控制语句:continue、break 和 return使用方法
- L017:continue和break - 教程文案
- 作为前端开发者,你都经历过怎样的面试?
- 面试被问 const 是否不可变?这样回答才显功底
- 标签列表
-
- git pull (33)
- git fetch (35)
- mysql insert (35)
- mysql distinct (37)
- concat_ws (36)
- java continue (36)
- jenkins官网 (37)
- mysql 子查询 (37)
- python元组 (33)
- mybatis 分页 (35)
- vba split (37)
- redis watch (34)
- python list sort (37)
- nvarchar2 (34)
- mysql not null (36)
- hmset (35)
- python telnet (35)
- python readlines() 方法 (36)
- munmap (35)
- docker network create (35)
- redis 集合 (37)
- python sftp (37)
- setpriority (34)
- c语言 switch (34)
- git commit (34)
