当前位置: 首页 > ds >正文

C#如何动态生成实体类?5种方法详解与实战演示

摘要:本文介绍C#中动态生成实体类的5种实用方法,涵盖T4模板、CodeDOM、Roslyn、反射和Emit等技术,通过真实代码示例帮助开发者应对不同场景需求。


一、应用场景分析

动态生成实体类常用于:

  • 数据库表结构变动时自动同步

  • 动态解析JSON/XML等异构数据源

  • 减少重复编码工作

  • 运行时动态类型创建


二、实现方案对比

方法易用性灵活性性能适用阶段
T4模板★★★★★★设计时
CodeDOM★★★★★★设计/运行时
Roslyn API★★★★★★★运行时
Reflection.Emit★★★★★★运行时
第三方库★★★★★★★运行时

三、具体实现方法

方法1:使用T4模板生成(设计时)

<#@ template debug="false" hostspecific="true" language="C#" #>
<#@ output extension=".cs" #>
<#var className = "DynamicEntity";var properties = new Dictionary<string, string> {{"Id", "int"},{"Name", "string"}};
#>
// Auto-generated class
public class <#= className #>
{
<# foreach(var prop in properties) { #>public <#= prop.Value #> <#= prop.Key #> { get; set; }
<# } #>
}

优点:Visual Studio原生支持
缺点:需要预生成文件


方法2:使用CodeDOM(运行时)

var compileUnit = new CodeCompileUnit();
var @namespace = new CodeNamespace("DynamicEntities");
var @class = new CodeTypeDeclaration("Person") { IsClass = true };@class.Members.Add(new CodeMemberField(typeof(int), "_age"));
var property = new CodeMemberProperty()
{Name = "Age",Type = new CodeTypeReference(typeof(int)),Attributes = MemberAttributes.Public
};
property.GetStatements.Add(new CodeMethodReturnStatement(new CodeFieldReferenceExpression(null, "_age")));
property.SetStatements.Add(new CodeAssignStatement(new CodeFieldReferenceExpression(null, "_age"),new CodePropertySetValueReferenceExpression()));@class.Members.Add(property);
var provider = new CSharpCodeProvider();
using var writer = new StringWriter();
provider.GenerateCodeFromCompileUnit(compileUnit, writer, null);
File.WriteAllText("Person.cs", writer.ToString());

适用场景:需要生成完整类文件时


方法3:使用Roslyn API(运行时)

var syntaxTree = CSharpSyntaxTree.ParseText(@"
using System;
namespace DynamicEntities
{public class Employee{public string FirstName { get; set; }public decimal Salary { get; set; }}
}");var compilation = CSharpCompilation.Create("DynamicAssembly").AddReferences(MetadataReference.CreateFromFile(typeof(object).Assembly.Location)).AddSyntaxTrees(syntaxTree);using var ms = new MemoryStream();
var result = compilation.Emit(ms);
if (result.Success)
{var assembly = Assembly.Load(ms.ToArray());dynamic obj = Activator.CreateInstance(assembly.GetType("DynamicEntities.Employee"));obj.FirstName = "John";
}

优势:支持完整的编译流程
注意:需要安装Microsoft.CodeAnalysis.CSharp包


方法4:使用Reflection.Emit(高性能)

var assemblyName = new AssemblyName("DynamicAssembly");
var assemblyBuilder = AssemblyBuilder.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
var moduleBuilder = assemblyBuilder.DefineDynamicModule("MainModule");
var typeBuilder = moduleBuilder.DefineType("Product", TypeAttributes.Public);// 添加属性
var fieldBuilder = typeBuilder.DefineField("_price", typeof(decimal), FieldAttributes.Private);
var propertyBuilder = typeBuilder.DefineProperty("Price", PropertyAttributes.None, typeof(decimal), null);// 生成get/set方法
var getSetAttr = MethodAttributes.Public | MethodAttributes.SpecialName;
var getMethod = typeBuilder.DefineMethod("get_Price", getSetAttr, typeof(decimal), Type.EmptyTypes);
var getIL = getMethod.GetILGenerator();
getIL.Emit(OpCodes.Ldarg_0);
getIL.Emit(OpCodes.Ldfld, fieldBuilder);
getIL.Emit(OpCodes.Ret);var setMethod = typeBuilder.DefineMethod("set_Price", getSetAttr, null, new[] { typeof(decimal) });
var setIL = setMethod.GetILGenerator();
setIL.Emit(OpCodes.Ldarg_0);
setIL.Emit(OpCodes.Ldarg_1);
setIL.Emit(OpCodes.Stfld, fieldBuilder);
setIL.Emit(OpCodes.Ret);propertyBuilder.SetGetMethod(getMethod);
propertyBuilder.SetSetMethod(setMethod);var dynamicType = typeBuilder.CreateType();
dynamic obj = Activator.CreateInstance(dynamicType);
obj.Price = 99.99m;

特点:最高性能,适合高频使用场景


四、注意事项

  1. 动态程序集无法卸载问题

  2. 类型冲突处理

  3. 调试困难建议添加异常处理

  4. 考虑使用缓存机制提升性能


五、方案选型建议

  • 简单场景:选择T4模板

  • 需要动态编译:使用Roslyn

  • 高性能需求:优先Emit

  • 快速开发:选择第三方库如Newtonsoft.Json


结语:根据项目需求选择合适方案,建议从T4模板开始熟悉,逐步掌握Emit等高级技巧。欢迎在评论区交流实际应用场景!


推荐工具

  • LINQPad:快速测试动态代码

  • ILSpy:查看生成的IL代码

  • Microsoft.CodeAnalysis:Roslyn核心库

http://www.xdnf.cn/news/317.html

相关文章:

  • 运营商二要素API接口有哪些应用场景?
  • 2025年03月中国电子学会青少年软件编程(Python)等级考试试卷(三级)真题
  • 练习(杨辉三角、字符串旋转)
  • 算法-链表
  • datasheet数据手册-阅读方法
  • 拆机装机,通电主板亮灯风扇不转无法开机解决办法
  • Three.js 教程导航页【目录总览】
  • SpringBoot集成oshi 查询系统数据
  • EVAL长度限制突破
  • 2000V高压储能趋势下,磁环如何实现高低温衰减≤12%
  • 性能测试面试题的详细解答
  • 习题3.4 最长连续递增子序列
  • 习题2.8 输出全排列
  • yarn常用的命令
  • Spring_MVC 快速入门指南
  • C++程序设计基础实验:C++对C的扩展特性与应用
  • Libtorch安装与使用
  • kubernetes-使用ceph-csi
  • Linux 进程
  • 浅析MySQL事务锁
  • 全球IP地址查询API:查询该IP所属的区域,提供即时服务
  • 【时时三省】(C语言基础)用do...while语句实现循环
  • 云端免费训练 AI 大模型推荐(适用于个人学习)
  • github 项目迁移到 gitee
  • 多肽 N 端乙酰化有什么作用?
  • 【机试】高精度
  • Java 数据类型全解析:基础、引用与包装类全面梳理
  • 【杂说杂谈】国产替代计划的思考
  • 使用droidrun库实现AI控制安卓手机
  • C++17 信号量模拟实现