.NET Emit 入門教程:第六部分:IL 指令:5:詳解 ILGenerator 指令方法:創建實例指令

来源:https://www.cnblogs.com/cyq1162/p/18112967
-Advertisement-
Play Games

在.NET Emit 入門教程的第六部分中,我們深入探討了 ILGenerator 指令方法,特別是關於創建實例指令的詳細解釋。ILGenerator 是.NET框架中的一個強大工具,用於在運行時生成和執行IL代碼。在這篇文章中,我們學習瞭如何使用 ILGenerator 來創建實例,其中主要涉及到... ...


前言:

上上篇介紹了 IL 指令的分類以及參數載入指令,該載入指令以 Ld 開頭,將參數載入到棧中,以便於後續執行操作命令。

上一篇介紹參數存儲指令,其指令以 St 開頭,將棧中的數據,存儲到指定的變數中,以方便後續使用。

本篇將介紹創建實例指令,其指令以 New 開頭,用於在運行時動態生成並初始化對象。

創建實例指令簡介

在.NET Emit 中,使用 ILGenerator 創建實例是一項重要的操作,它允許我們動態生成對象實例和數組實例的代碼。

通過創建實例指令,我們可以在運行時動態生成並初始化對象,為程式提供更大的靈活性和可擴展性。

創建實例指令主要包括 Newobj 指令和 Newarr 指令。

Newobj 指令用於創建新的對象實例,而 Newarr 指令則用於創建新的數組實例。

這些指令的靈活運用可以幫助我們在運行時動態地生成各種類型的實例,滿足不同場景下的需求。

在本篇文章中,我們將深入探討 ILGenerator 中的創建實例指令,詳細解析其用法和示例代碼。

通過學習本文內容,讀者將能夠掌握如何利用 ILGenerator 創建對象實例和數組實例,從而更好地理解和應用.NET Emit 技術。

1、創建實例指令:Newobj

對於該指令,其核心在於如何獲取構造函數並作為參數傳遞,下麵看一組示例。

共用代碼,定義實體(包含無參構造函數、有參構造函數、基本變數):

 public class Entity
 {
     public Entity()
     {

     }
     public Entity(int id)
     {
         this.ID = id;
     }
     public int ID;
 }

共用代碼,生成程式集,以方便後續對照參考:

AssemblyName assName = new AssemblyName("myAssembly") { Version = new Version("1.1.1.2") };
AssemblyBuilder ab = AppDomain.CurrentDomain.DefineDynamicAssembly(assName, AssemblyBuilderAccess.RunAndSave);
ModuleBuilder mb = ab.DefineDynamicModule("myModule", "b.dll");
TypeBuilder tb = mb.DefineType("MyNameSpace.MyClass", TypeAttributes.Public | TypeAttributes.Class);

//定義靜態方法
MethodBuilder methodBuilder = tb.DefineMethod("NewObj", MethodAttributes.Public | MethodAttributes.Static, typeof(Entity), new Type[] { });
ILGenerator il = methodBuilder.GetILGenerator();

//il 代碼處......


Type classType = tb.CreateType();

ab.Save("b.dll");

A、無參數實例化:通過 Type 的 GetConstructor 實例方法獲取類型的構造函數。

ILGenerator il = methodBuilder.GetILGenerator();
il.Emit(OpCodes.Newobj, typeof(Entity).GetConstructor(Type.EmptyTypes));
il.Emit(OpCodes.Ret);     // 返回該值

對照生成:

B、使用參數實例化:

 ILGenerator il = methodBuilder.GetILGenerator();
 il.Emit(OpCodes.Ldc_I4, 999);
 il.Emit(OpCodes.Newobj, typeof(Entity).GetConstructor(BindingFlags.Public | BindingFlags.Instance, null, new Type[] { typeof(int) }, null));
 il.Emit(OpCodes.Ret);     // 返回該值

對照生成:

小說明:

這裡構造函數的參數傳入,是通過 Ld 系列指令按順序壓入棧中。

2、創建實例指令:Newarr

該指令用於創建數組對象,該指令需要指定數組長度。

A、創建數組:

ILGenerator il = methodBuilder.GetILGenerator();
il.Emit(OpCodes.Ldc_I4, 6);
il.Emit(OpCodes.Newarr, typeof(Entity));
il.Emit(OpCodes.Ret);     // 返回該值

對照生成代碼:

小說明:

Newarr 接收的參數,是 Type 類型。

Newobj 接收的參數,是 ConstructorInfo 構造函數類型。

B、對數組賦值:引用類型

ILGenerator il = methodBuilder.GetILGenerator();

//創建數組
il.Emit(OpCodes.Ldc_I4, 3);
il.Emit(OpCodes.Newarr, typeof(Entity));

il.DeclareLocal(typeof(Entity[]));
il.DeclareLocal(typeof(Entity));
il.Emit(OpCodes.Stloc_0);//存儲數組

for (int i = 0; i < 3; i++)
{
    il.Emit(OpCodes.Newobj, typeof(Entity).GetConstructor(Type.EmptyTypes));//定義實體類
    il.Emit(OpCodes.Stloc_1);//存儲實體類

    il.Emit(OpCodes.Ldloc_0);//載入數組
    il.Emit(OpCodes.Ldc_I4, i);//載入索引
    il.Emit(OpCodes.Ldloc_1);//載入Entity

    il.Emit(OpCodes.Stelem_Ref);//引用類型賦值
}


il.Emit(OpCodes.Ldloc_0);//載入數組
il.Emit(OpCodes.Ret);     // 返回該值

對照生成代碼:

C、對數組賦值:值類型

ILGenerator il = methodBuilder.GetILGenerator();

//創建數組
il.Emit(OpCodes.Ldc_I4, 3);
il.Emit(OpCodes.Newarr, typeof(DateTime));

il.DeclareLocal(typeof(DateTime[]));
il.DeclareLocal(typeof(DateTime));
il.Emit(OpCodes.Stloc_0);

for (int i = 0; i < 3; i++)
{// 調用 DateTime.Parse 方法創建 DateTime 實例
    MethodInfo parseMethod = typeof(DateTime).GetMethod("Parse", new Type[] { typeof(string) });
    il.Emit(OpCodes.Ldstr, DateTime.Now.ToString()); // 傳遞當前時間字元串
    il.Emit(OpCodes.Call, parseMethod);    // 調用 Parse 方法
    il.Emit(OpCodes.Stloc_1);

    il.Emit(OpCodes.Ldloc_0);//載入數組
    il.Emit(OpCodes.Ldc_I4, i);//載入索引
    il.Emit(OpCodes.Ldloc_1);//載入Entity


    il.Emit(OpCodes.Stelem, typeof(DateTime));//賦值
}

il.Emit(OpCodes.Ldloc_0);
il.Emit(OpCodes.Ret);     // 返回該值

對照生成代碼:

D、數組取值指令:

總結:

在.NET Emit 入門教程的第六部分中,我們深入探討了 ILGenerator 指令方法,特別是關於創建實例指令的詳細解釋。

ILGenerator 是.NET框架中的一個強大工具,用於在運行時生成和執行IL代碼。

在這篇文章中,我們學習瞭如何使用 ILGenerator 來創建實例,其中主要涉及到了兩種指令方法:newobj 和 newarr。

通過 newobj 指令,我們可以在IL代碼中調用構造函數來創建類的實例,而 newarr 指令則用於創建數組實例。

通過學習這些內容,讀者可以更深入地理解 ILGenerator 的使用,併在實際項目中應用動態代碼生成的技術。

下一篇,我們將學習方法調用指令的相關內容。

版權聲明:本文原創發表於 博客園,作者為 路過秋天 本文歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則視為侵權。
個人微信公眾號
Donation(掃碼支持作者):支付寶:
Donation(掃碼支持作者):微信:

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • C++ 20 的 std::format 是一個很神奇、很實用的工具,最神奇的地方在於它能在編譯期檢查字元串的格式是否正確,而且不需要什麼特殊的使用方法,只需要像使用普通函數那樣傳參即可。 #include <format> int a = 1; std::string s1 = std::form ...
  • 官網:Vue Router | Vue.js 的官方路由 (vuejs.org) 安裝命令:npm install vue-router@4 1.添加兩個頁面\vuedemo\src\views\index.vue、\vuedemo\src\views\content.vue 2.添加\vuedem ...
  • 首先插件添加:Live Server、Vue - Official、Vue VSCode Snippets、別名路徑跳轉 官網:Vite | 下一代的前端工具鏈 (vitejs.dev) 1.創建一個文件夾VueApp,運行cmd轉到該目錄下,執行命令:npm create vite@latest ...
  • 1.安裝node.js : Node.js — Run JavaScript Everywhere (nodejs.org) 2.查看安裝版本命令:node -v 、 npm -v, npm是Node.js包管理器, 用來安裝各種庫、框架和工具。 3.查看當前的鏡像源: npm get regist ...
  • 在Avalonia的UI框架中,TemplatedControl是一個核心組件,它提供了一種強大的方式來創建可重用且高度可定製的控制項。 本文將深入探討TemplatedControl的概念、其帶來的優勢以及它在實際開發中的應用場景,並通過一個示例代碼來展示其用法。 什麼是TemplatedContr ...
  • 本文探討了.NET Emit 入門教程的第六部分,聚焦於ILGenerator中的方法調用指令。通過詳細分析 ILGenerator 的使用方法和方法調用指令,讀者可以更深入地瞭解.NET平臺下動態生成代碼的實現機制。通過本文的閱讀,讀者可以更加熟練地使用 ILGenerator 來動態生成高效、靈... ...
  • 自上一個系列文章寫完之後,最近的兩三個月時間,一直在寫WPF相關技術文章,斷斷續續地寫了近二十幾篇文章,為了方便大家閱讀,現將本系列文章分類整理,彙總如下。 ...
  • 今天給大家上個硬貨,下拉多選框,同時也是下拉多選樹,支持父節點跟子節點!該控制項是基於Telerik控制項封裝實現的,所以大家在使用的過程中需要引用Telerik.WinControls.dll、Telerik.WinControls.UI.dll,還有一些相關的類庫,大家有需要的可以去網上自己找,另外 ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...