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

 更新时间:2025年04月24日 10:33:50   作者:LcVong  
这篇文章主要为大家详细介绍了C#中动态生成实体类的5种实用方法,涵盖T4模板,CodeDOM,Roslyn,反射和Emit等技术,有需要的小伙伴可以跟随小编一起学习一下

摘要:本文介绍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;

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

四、注意事项

动态程序集无法卸载问题

类型冲突处理

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

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

五、方案选型建议

简单场景:选择T4模板

需要动态编译:使用Roslyn

高性能需求:优先Emit

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

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

推荐工具:

LINQPad:快速测试动态代码

ILSpy:查看生成的IL代码

Microsoft.CodeAnalysis:Roslyn核心库

到此这篇关于C#动态生成实体类的5种方法详解与实战演示的文章就介绍到这了,更多相关C#动态生成实体类内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • DevExpress实现自定义GridControl中按钮文字内容的方法

    DevExpress实现自定义GridControl中按钮文字内容的方法

    这篇文章主要介绍了DevExpress实现自定义GridControl中按钮文字内容的方法,需要的朋友可以参考下
    2014-08-08
  • 详解C++中类的大小决定因数

    详解C++中类的大小决定因数

    类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下
    2025-04-04
  • c#索引器详解示例

    c#索引器详解示例

    索引器是一种特殊的类成员,它能够让对象以类似数组的方式来存取,使程序看起来更为直观,更容易编写。
    2014-04-04
  • WPF仿三星手机充电界面实现代码

    WPF仿三星手机充电界面实现代码

    这篇文章主要为大家详细介绍了WPF仿三星手机充电界面实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-08-08
  • C#利用SharpPcap实现网络包捕获嗅探

    C#利用SharpPcap实现网络包捕获嗅探

    这篇文章主要为大家详细介绍了C#利用SharpPcap实现网络包捕获嗅探,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-03-03
  • C# 在PDF文档中创建表格的实现方法

    C# 在PDF文档中创建表格的实现方法

    表格能够一目了然的让用户看到数据信息,使信息显得有条理化,那么在pdf类型的文档中如何来添加表格并对表格进行格式化操作呢?下面小编给大家带来了C# 在PDF文档中创建表格的实现方法,需要的朋友参考下吧
    2017-12-12
  • 通过C#实现发送自定义的html格式邮件

    通过C#实现发送自定义的html格式邮件

    本篇文章主要介绍了通过C#实现发送自定义的html格式邮件,详细的介绍了发送HTML格式邮件的方法,有兴趣的可以了解一下。
    2017-02-02
  • 解析C#拼接Json串的几种方法

    解析C#拼接Json串的几种方法

    本文主要介绍了C#拼接Json串的几种方法,主要包括了4种方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-06-06
  • C#实现时间戳与标准时间的互转

    C#实现时间戳与标准时间的互转

    本文主要介绍了C#中时间戳与标准时间互转的方法,其中需要注意的是基准时间的问题。文中的示例代码具有一定的学习价值,快来跟随小编一起了解一下吧
    2021-12-12
  • C#实现提高xml读写速度的方法

    C#实现提高xml读写速度的方法

    这篇文章主要介绍了C#实现提高xml读写速度的方法,并且针对各类文件的读写做了较为细致的分析,非常实用,需要的朋友可以参考下
    2014-11-11

最新评论