c# 在Emit代碼中如何await一個(gè)異步方法
0. 前言
首先立馬解釋一波為啥會(huì)有這樣一篇偽標(biāo)題的Demo隨筆呢?
不是本人有知識(shí)誤區(qū),或者要誤人子弟
因?yàn)榇蠹叶贾纄mit寫(xiě)出來(lái)的都是同步方法,不可能await,至少現(xiàn)在這么多年來(lái)沒(méi)有提供對(duì)應(yīng)的功能
這是之前某天在微信群看見(jiàn)討論怎么emit一個(gè)異步方法并包裝異步結(jié)構(gòu),簡(jiǎn)單幾句文字也未能清晰的表達(dá)
所以趁著元旦節(jié)放假有點(diǎn)時(shí)間,
簡(jiǎn)單列舉三種我知道方式去達(dá)到這樣的效果
三種方法都是繞過(guò)emit直接書(shū)寫(xiě)emit代碼,而是將對(duì)應(yīng)邏輯轉(zhuǎn)到其他方法中,最后emit調(diào)用方法達(dá)到效果
Demo 說(shuō)明
原始方法是個(gè)延遲2秒之后返回55的方法:
public static async Task<int> GetV()
{
await Task.Delay(2000);
return 55;
}
現(xiàn)在我們需要把 55 的結(jié)果加 6 ,讓最終的結(jié)果變?yōu)?61
我們的測(cè)試方法是這樣,會(huì)輸出一些簡(jiǎn)單的時(shí)間,幫助我們了解執(zhí)行順序和異步情況
private static async Task Test(MethodInfo method, MethodInfo awaitMehtod)
{
var caller = CreateCaller(method, awaitMehtod);
Console.WriteLine($"Start {awaitMehtod.Name} at: {DateTime.Now}.");
var task = caller();
Console.WriteLine($"Call done at: {DateTime.Now}.");
var number = await task;
Console.WriteLine($"Hello {number} at: {DateTime.Now}.");
Console.WriteLine($"End at: {DateTime.Now}.");
Console.WriteLine();
}
1. ContinueWith
public static Func<Task<int>> CreateCaller(MethodInfo method, MethodInfo awaitMehtod)
{
var m = new DynamicMethod(Guid.NewGuid().ToString("N"), typeof(Task<int>), Type.EmptyTypes);
var il = m.GetILGenerator();
il.Emit(OpCodes.Call, method);
il.Emit(OpCodes.Call, typeof(Program).GetMethod(nameof(Program.AddSixUseContinueWith))); // 這里是差異點(diǎn)
il.Emit(OpCodes.Ret);
return m.CreateDelegate(typeof(Func<Task<int>>)) as Func<Task<int>>;
}
public static Task<int> AddSixUseContinueWith(Task<int> task)
{
return task.ContinueWith(i =>
{
Console.WriteLine($"AddSixUseContinueWith is: {DateTime.Now}.");
return i.Result + 6;
});
}
測(cè)試結(jié)果:
Start AddSixUseContinueWith at: 2021/1/2 13:34:55. Call done at: 2021/1/2 13:34:55. AddSixUseContinueWith is: 2021/1/2 13:34:57. Hello 61 at: 2021/1/2 13:34:57. End at: 2021/1/2 13:34:57.
優(yōu)點(diǎn)
還是真正的異步
缺點(diǎn)
成本比較大,畢竟這樣沒(méi)有了狀態(tài)機(jī)等等優(yōu)化,(成本在 ns 級(jí)別哦,不是大家想的 ms哦)
2. GetAwaiter().GetResult()
public static Func<Task<int>> CreateCaller(MethodInfo method, MethodInfo awaitMehtod)
{
var m = new DynamicMethod(Guid.NewGuid().ToString("N"), typeof(Task<int>), Type.EmptyTypes);
var il = m.GetILGenerator();
il.Emit(OpCodes.Call, method);
il.Emit(OpCodes.Call, typeof(Program).GetMethod(nameof(Program.AddSixUseAwaiter))); // 這里是差異點(diǎn)
il.Emit(OpCodes.Ret);
return m.CreateDelegate(typeof(Func<Task<int>>)) as Func<Task<int>>;
}
public static Task<int> AddSixUseAwaiter(Task<int> task)
{
var r = task.ConfigureAwait(false).GetAwaiter().GetResult() + 6;
Console.WriteLine($"AddSixUseAwaiter is: {DateTime.Now}.");
return Task.FromResult(r);
}
測(cè)試結(jié)果:
Start AddSixUseAwaiter at: 2021/1/2 13:34:57. AddSixUseAwaiter is: 2021/1/2 13:34:59. Call done at: 2021/1/2 13:34:59. Hello 61 at: 2021/1/2 13:34:59. End at: 2021/1/2 13:34:59.
優(yōu)點(diǎn)
執(zhí)行時(shí)間上消耗很小
缺點(diǎn)
當(dāng)然這樣 異步都變成了同步,所以可能會(huì)在某些情況下我們操作不當(dāng)?shù)拇a從而導(dǎo)致失去異步方法的優(yōu)勢(shì)
3. async/await
public static Func<Task<int>> CreateCaller(MethodInfo method, MethodInfo awaitMehtod)
{
var m = new DynamicMethod(Guid.NewGuid().ToString("N"), typeof(Task<int>), Type.EmptyTypes);
var il = m.GetILGenerator();
il.Emit(OpCodes.Call, method);
il.Emit(OpCodes.Call, typeof(Program).GetMethod(nameof(Program.AddSixUseAsyncAwait))); // 這里是差異點(diǎn)
il.Emit(OpCodes.Ret);
return m.CreateDelegate(typeof(Func<Task<int>>)) as Func<Task<int>>;
}
public static async Task<int> AddSixUseAsyncAwait(Task<int> task)
{
var r = await task;
Console.WriteLine($"AddSixUseAsyncAwait is: {DateTime.Now}.");
return r + 6;
}
測(cè)試結(jié)果:
Start AddSixUseAsyncAwait at: 2021/1/2 13:34:59. Call done at: 2021/1/2 13:34:59. AddSixUseAsyncAwait is: 2021/1/2 13:35:01. Hello 61 at: 2021/1/2 13:35:01. End at: 2021/1/2 13:35:01.
優(yōu)點(diǎn)
async / await 本身的優(yōu)勢(shì)都沒(méi)有損失
缺點(diǎn)
原本想在 emit 中 對(duì)result的處理邏輯 必須遷移到 async / await 方法中,emit代碼必須好好設(shè)計(jì)
完整Demo放在
https://github.com/fs7744/grocery/blob/main/csharp/emit_await/EmitAwaitDemo/Program.cs
分享不易,如果能給予一點(diǎn)動(dòng)力,不勝感激:關(guān)注一下本人的開(kāi)源項(xiàng)目: Norns.Urd
以上就是c# 在Emit代碼中如何await一個(gè)異步方法的詳細(xì)內(nèi)容,更多關(guān)于c# Emit代碼await一個(gè)異步方法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C#?HttpClient超時(shí)重試機(jī)制詳解
超時(shí)重試的實(shí)現(xiàn)方式可以使用循環(huán)結(jié)構(gòu),在請(qǐng)求發(fā)起后等待一定時(shí)間,若超時(shí)未收到響應(yīng),則再次發(fā)起請(qǐng)求,循環(huán)次數(shù)可以根據(jù)實(shí)際情況進(jìn)行設(shè)置,一般建議不超過(guò)三次,這篇文章主要介紹了C#?HttpClient超時(shí)重試,需要的朋友可以參考下2023-06-06
C#實(shí)現(xiàn)Quartz任務(wù)調(diào)度的示例代碼
使用 Quartz.NET,你可以很容易地安排任務(wù)在應(yīng)用程序啟動(dòng)時(shí)運(yùn)行,或者每天、每周、每月的特定時(shí)間運(yùn)行,甚至可以基于更復(fù)雜的調(diào)度規(guī)則,本文給大家介紹了C#實(shí)現(xiàn)Quartz任務(wù)調(diào)度,需要的朋友可以參考下2024-04-04
C#實(shí)現(xiàn)Winform中打開(kāi)網(wǎng)頁(yè)頁(yè)面的方法
這篇文章主要介紹了C#實(shí)現(xiàn)Winform中打開(kāi)網(wǎng)頁(yè)頁(yè)面的方法,涉及WinForm中WebBrowser的相關(guān)使用技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-08-08
詳解C# parallel中并行計(jì)算的四種寫(xiě)法總結(jié)
在C#中,parallel關(guān)鍵字可以用于并行計(jì)算。本文為大家總結(jié)了四種C# parallel并行計(jì)算的方法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-11-11
C#使用jQuery實(shí)現(xiàn)無(wú)刷新評(píng)論提交的方法
這篇文章主要介紹了C#使用jQuery實(shí)現(xiàn)無(wú)刷新評(píng)論提交的方法,涉及C#結(jié)合jQuery進(jìn)行Ajax操作的相關(guān)技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-05-05
C# 輸出字符串到文本文件中的實(shí)現(xiàn)代碼
本文通過(guò)一個(gè)簡(jiǎn)單的代碼給大家介紹C# 輸出字符串到文本文件中,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2018-05-05

