欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C#動態(tài)生成實體類的5種方法詳解與實戰(zhàn)演示

 更新時間:2025年04月24日 10:33:50   作者:LcVong  
這篇文章主要為大家詳細(xì)介紹了C#中動態(tài)生成實體類的5種實用方法,涵蓋T4模板,CodeDOM,Roslyn,反射和Emit等技術(shù),有需要的小伙伴可以跟隨小編一起學(xué)習(xí)一下

摘要:本文介紹C#中動態(tài)生成實體類的5種實用方法,涵蓋T4模板、CodeDOM、Roslyn、反射和Emit等技術(shù),通過真實代碼示例幫助開發(fā)者應(yīng)對不同場景需求。

一、應(yīng)用場景分析

動態(tài)生成實體類常用于:

數(shù)據(jù)庫表結(jié)構(gòu)變動時自動同步

動態(tài)解析JSON/XML等異構(gòu)數(shù)據(jù)源

減少重復(fù)編碼工作

運行時動態(tài)類型創(chuàng)建

二、實現(xiàn)方案對比

方法易用性靈活性性能適用階段
T4模板★★★★★★設(shè)計時
CodeDOM★★★★★★設(shè)計/運行時
Roslyn API★★★★★★★運行時
Reflection.Emit★★★★★★運行時
第三方庫★★★★★★★運行時

三、具體實現(xiàn)方法

方法1:使用T4模板生成(設(shè)計時)

<#@ 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; }
<# } #>
}

優(yōu)點:Visual Studio原生支持

缺點:需要預(yù)生成文件

方法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";
}

優(yōu)勢:支持完整的編譯流程

注意:需要安裝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;

特點:最高性能,適合高頻使用場景

四、注意事項

動態(tài)程序集無法卸載問題

類型沖突處理

調(diào)試?yán)щy建議添加異常處理

考慮使用緩存機制提升性能

五、方案選型建議

簡單場景:選擇T4模板

需要動態(tài)編譯:使用Roslyn

高性能需求:優(yōu)先Emit

快速開發(fā):選擇第三方庫如Newtonsoft.Json

結(jié)語:根據(jù)項目需求選擇合適方案,建議從T4模板開始熟悉,逐步掌握Emit等高級技巧。歡迎在評論區(qū)交流實際應(yīng)用場景!

推薦工具:

LINQPad:快速測試動態(tài)代碼

ILSpy:查看生成的IL代碼

Microsoft.CodeAnalysis:Roslyn核心庫

到此這篇關(guān)于C#動態(tài)生成實體類的5種方法詳解與實戰(zhàn)演示的文章就介紹到這了,更多相關(guān)C#動態(tài)生成實體類內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評論