校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識(shí)庫 > .Net core 的熱插拔機(jī)制的深入探索及卸載問題求救指南

.Net core 的熱插拔機(jī)制的深入探索及卸載問題求救指南

熱門標(biāo)簽:許昌智能電銷機(jī)器人公司 遼寧正規(guī)電銷機(jī)器人 辰溪地圖標(biāo)注 澳大利亞城市地圖標(biāo)注 遼寧銀行智能外呼系統(tǒng) 上海浦東騰訊地圖標(biāo)注位置 電銷機(jī)器人違法了嗎 海南銀行智能外呼系統(tǒng)商家 姜堰電銷機(jī)器人

一.依賴文件*.deps.json的讀取.

依賴文件內(nèi)容如下.一般位于編譯生成目錄中

{
 "runtimeTarget": {
 "name": ".NETCoreApp,Version=v3.1",
 "signature": ""
 },
 "compilationOptions": {},
 "targets": {
 ".NETCoreApp,Version=v3.1": {
 "PluginSample/1.0.0": {
 "dependencies": {
 "Microsoft.Extensions.Hosting.Abstractions": "5.0.0-rc.2.20475.5"
 },
 "runtime": {
 "PluginSample.dll": {}
 }
 },
 "Microsoft.Extensions.Configuration.Abstractions/5.0.0-rc.2.20475.5": {
 "dependencies": {
 "Microsoft.Extensions.Primitives": "5.0.0-rc.2.20475.5"
 },
 "runtime": {
 "lib/netstandard2.0/Microsoft.Extensions.Configuration.Abstractions.dll": {
 "assemblyVersion": "5.0.0.0",
 "fileVersion": "5.0.20.47505"
 }
 }
 ...

使用DependencyContextJsonReader加載依賴配置文件源碼查看

using (var dependencyFileStream = File.OpenRead("Sample.deps.json"))
{
 using (DependencyContextJsonReader dependencyContextJsonReader = new DependencyContextJsonReader())
 {
 //得到對應(yīng)的實(shí)體文件
 var dependencyContext = 
 dependencyContextJsonReader.Read(dependencyFileStream);
 //定義的運(yùn)行環(huán)境,沒有,則為全平臺(tái)運(yùn)行.
 string currentRuntimeIdentifier= dependencyContext.Target.Runtime;
 //運(yùn)行時(shí)所需要的dll文件
 var assemblyNames= dependencyContext.RuntimeLibraries;
 }
}

二.Net core多平臺(tái)下RID(RuntimeIdentifier)的定義.

安裝 Microsoft.NETCore.Platforms包,并找到runtime.json運(yùn)行時(shí)定義文件.

{
 "runtimes": {
 "win-arm64": {
 "#import": [
 "win"
 ]
 },
 "win-arm64-aot": {
 "#import": [
 "win-aot",
 "win-arm64"
 ]
 },
 "win-x64": {
 "#import": [
 "win"
 ]
 },
 "win-x64-aot": {
 "#import": [
 "win-aot",
 "win-x64"
 ]
 },
}

NET Core RID依賴關(guān)系示意圖

win7-x64 win7-x86
 | \ / |
 | win7 |
 | | |
win-x64 | win-x86
 \ | /
 win
 |
 any

.Net core常用發(fā)布平臺(tái)RID如下

  • windows (win)

win-x64
win-x32
win-arm

  • macos (osx)

osx-x64

  • linux (linux)

linux-x64
linux-arm

1. .net core的runtime.json文件由微軟提供:查看runtime.json.

2. runtime.json的runeims節(jié)點(diǎn)下,定義了所有的RID字典表以及RID樹關(guān)系.

3. 根據(jù)*.deps.json依賴文件中的程序集定義RID標(biāo)識(shí),就可以判斷出依賴文件中指向的dll是否能在某一平臺(tái)運(yùn)行.

4. 當(dāng)程序發(fā)布為兼容模式時(shí),我們出可以使用runtime.json文件選擇性的加載平臺(tái)dll并運(yùn)行.

三.AssemblyLoadContext的加載原理

public class PluginLoadContext : AssemblyLoadContext
{
 private AssemblyDependencyResolver _resolver;
 public PluginLoadContext(string pluginFolder, params string[] commonAssemblyFolders) : base(isCollectible: true)
 {
 this.ResolvingUnmanagedDll += PluginLoadContext_ResolvingUnmanagedDll;
 this.Resolving += PluginLoadContext_Resolving;
 //第1步,解析des.json文件,并調(diào)用Load和LoadUnmanagedDll函數(shù)
 _resolver = new AssemblyDependencyResolver(pluginFolder);
 //第6步,通過第4,5步,解析仍失敗的dll會(huì)自動(dòng)嘗試調(diào)用主程序中的程序集,
 //如果失敗,則直接拋出程序集無法加載的錯(cuò)誤
 }
 private Assembly PluginLoadContext_Resolving(AssemblyLoadContext assemblyLoadContext, AssemblyName assemblyName)
 {
 //第4步,Load函數(shù)加載程序集失敗后,執(zhí)行的事件
 }
 private IntPtr PluginLoadContext_ResolvingUnmanagedDll(Assembly assembly, string unmanagedDllName)
 {
 //第5步,LoadUnmanagedDll加載native dll失敗后執(zhí)行的事件
 }
 protected override Assembly Load(AssemblyName assemblyName)
 {
 //第2步,先執(zhí)行程序集的加載函數(shù)
 }
 protected override IntPtr LoadUnmanagedDll(string unmanagedDllName)
 {
 //第3步,先執(zhí)行的native dll加載邏輯
 }
}

微軟官方示例代碼如下:示例具體內(nèi)容

class PluginLoadContext : AssemblyLoadContext
{
 private AssemblyDependencyResolver _resolver;

 public PluginLoadContext(string pluginPath)
 {
 _resolver = new AssemblyDependencyResolver(pluginPath);
 }

 protected override Assembly Load(AssemblyName assemblyName)
 {
 string assemblyPath = _resolver.ResolveAssemblyToPath(assemblyName);
 if (assemblyPath != null)
 {
 //加載程序集
 return LoadFromAssemblyPath(assemblyPath);
 }
 //返回null,則直接加載主項(xiàng)目程序集
 return null;
 }

 protected override IntPtr LoadUnmanagedDll(string unmanagedDllName)
 {
 string libraryPath = _resolver.ResolveUnmanagedDllToPath(unmanagedDllName);
 if (libraryPath != null)
 {
 //加載native dll文件
 return LoadUnmanagedDllFromPath(libraryPath);
 }
 //返回IntPtr.Zero,即null指針.將會(huì)加載主項(xiàng)中runtimes文件夾下的dll
 return IntPtr.Zero;
 }
}

1. 官方這個(gè)示例是有問題的.LoadFromAssemblyPath()函數(shù)有bug,
該函數(shù)并不會(huì)加載依賴的程序集.正確用法是LoadFormStream()

2. Load和LoadUnmanagedDll函數(shù)實(shí)際上是給開發(fā)者手動(dòng)加載程序集使用的,
自動(dòng)加載應(yīng)放到Resolving和ResolvingUnmanagedDll事件中
原因是,這樣的加載順序不會(huì)導(dǎo)致項(xiàng)目的程序集覆蓋插件的程序集,造成程序集加載失敗.

3. 手動(dòng)加載時(shí)可以根據(jù)deps.json文件定義的runtime加載當(dāng)前平臺(tái)下的unmanaged dll文件.

這些平臺(tái)相關(guān)的dll文件,一般位于發(fā)布目錄中的runtimes文件夾中.

四.插件項(xiàng)目一定要和主項(xiàng)目使用同樣的運(yùn)行時(shí).

  1. 如果主項(xiàng)目是.net core 3.1,插件項(xiàng)目不能選擇.net core 2.0等,甚至不能選擇.net standard庫
  2. 否則會(huì)出現(xiàn)不可預(yù)知的問題.
  3. 插件是.net standard需要修改項(xiàng)目文件,TargetFrameworks>netstandard;netcoreapp3.1/TargetFrameworks>
  4. 這樣就可以發(fā)布為.net core項(xiàng)目.
  5. 若主項(xiàng)目中的nuget包不適合當(dāng)前平臺(tái),則會(huì)報(bào)Not Support Platform的異常.這時(shí)如果主項(xiàng)目是在windows上, 就需要把項(xiàng)目發(fā)布目標(biāo)設(shè)置為win-x64.這屬于nuget包依賴關(guān)系存在錯(cuò)誤描述.

五.AssemblyLoadContext.UnLoad()并不會(huì)拋出任何異常.

當(dāng)你調(diào)用AssemblyLoadContext.UnLoad()卸載完插件以為相關(guān)程序集已經(jīng)釋放,那你可能就錯(cuò)了.官方文檔表明卸載執(zhí)行失敗會(huì)拋出InvalidOperationException,不允許卸載官方說明。
但實(shí)際測試中,卸載失敗,但并未報(bào)錯(cuò).

六.反射程序集相關(guān)變量的定義為何阻止插件程序集卸載?

插件

namespace PluginSample
{
 public class SimpleService
 {
 public void Run(string name)
 {
 Console.WriteLine($"Hello World!");
 }
 }
}

加載插件

namespace Test
{
 public class PluginLoader
 {
 pubilc AssemblyLoadContext assemblyLoadContext;
 public Assembly assembly;
 public Type type;
 public MethodInfo method;
 public void Load()
 {
 assemblyLoadContext = new PluginLoadContext("插件文件夾");
 assembly = alc.Load(new AssemblyName("PluginSample"));
 type = assembly.GetType("PluginSample.SimpleService");
 method=type.GetMethod()
 }
 }
}

1. 在主項(xiàng)目程序中.AssemblyLoadContext,Assembly,Type,MethodInfo等不能直接定義在任何類中.
否則在插件卸載時(shí)會(huì)失敗.當(dāng)時(shí)為了測試是否卸載成功,采用手動(dòng)加載,執(zhí)行,卸載了1000次,
發(fā)現(xiàn)內(nèi)存一直上漲,則表示卸載失敗.

2. 參照官方文檔后了解了WeakReferece類.使用該類與AssemblyLoadContext關(guān)聯(lián),當(dāng)手動(dòng)GC清理時(shí),
AssemblyLoadContext就會(huì)變?yōu)閚ull值,如果沒有變?yōu)閚ull值則表示卸載失敗.

3. 使用WeakReference關(guān)聯(lián)AssemblyLoadContext并判斷是否卸載成功

public void Load(out WeakReference weakReference)
 {
 var assemblyLoadContext = new PluginLoadContext("插件文件夾");
 weakReference = new WeakReference(pluginLoadContext, true);
 assemblyLoadContext.UnLoad();
 }
 public void Check()
 {
 WeakReference weakReference=null;
 Load(out weakReference);
 //一般第二次,IsAlive就會(huì)變?yōu)镕alse,即AssemblyLoadContext卸載失敗.
 for (int i = 0; weakReference.IsAlive  (i  10); i++)
 {
 GC.Collect();
 GC.WaitForPendingFinalizers();
 }
 }

4. 為了解決以上問題.可以把需要的變量放到靜態(tài)字典中.在Unload之前把對應(yīng)的Key值刪除掉,即可.

七.程序集的異步函數(shù)執(zhí)行為何會(huì)阻止插件程序的卸載?

public class SimpleService
{
 //同步執(zhí)行,插件卸載成功
 public void Run(string name)
 {
 Console.WriteLine($"Hello {name}!");
 }
 //異步執(zhí)行,卸載成功
 public Task RunAsync(string name)
 {
 Console.WriteLine($"Hello {name}!");
 return Task.CompletedTask;
 }
 //異步執(zhí)行,卸載成功
 public Task RunTask(string name)
 {
 return Task.Run(() => {
 Console.WriteLine($"Hello {name}!");
 });
 }
 //異步執(zhí)行,卸載成功
 public Task RunWaitTask(string name)
 {
 return Task.Run( async ()=> {
 while (true)
 {
 if (CancellationTokenSource.IsCancellationRequested)
 {
  break;
 }
 await Task.Delay(1000);
 Console.WriteLine($"Hello {name}!");
 }
 });
 }
 //異步執(zhí)行,卸載成功
 public Task RunWaitTaskForCancel(string name, CancellationToken cancellation)
 {
 return Task.Run(async () => {
 while (true)
 {
 if (cancellation.IsCancellationRequested)
 {
  break;
 }
 await Task.Delay(1000);
 Console.WriteLine($"Hello {name}!");
 }
 });
 }
 //異步執(zhí)行,卸載失敗
 public async Task RunWait(string name)
 {
 while (true)
 {
 if (CancellationTokenSource.IsCancellationRequested)
 {
 break;
 }
 await Task.Delay(1000);
 Console.WriteLine($"Hello {name}!");
 }

 }
 //異步執(zhí)行,卸載失敗
 public Task RunWaitNewTask(string name)
 {
 return Task.Factory.StartNew(async ()=> {
 while (true)
 {
 if (CancellationTokenSource.IsCancellationRequested)
 {
  break;
 }
 await Task.Delay(1000);
 Console.WriteLine($"Hello {name}!");
 }
 },TaskCreationOptions.DenyChildAttach);
 }
}

1. 以上測試可以看出,如果插件調(diào)用的是一個(gè)常規(guī)帶wait的async異步函數(shù),則插件一定會(huì)卸載失敗.
原因推測是返回的結(jié)果是編譯器自動(dòng)生成的狀態(tài)機(jī)實(shí)現(xiàn)的,而狀態(tài)機(jī)是在插件中定義的.

2. 如果在插件中使用Task.Factory.StartNew函數(shù)也會(huì)調(diào)用失敗,原因不明.
官方文檔說和Task.Run函數(shù)是Task.Factory.StartNew的簡單形式,只是參數(shù)不同.官方說明
按照官方提供的默認(rèn)參數(shù)測試,卸載仍然失敗.說明這兩種方式實(shí)現(xiàn)底層應(yīng)該是不同的.

八.正確卸載插件的方式

  • 任何與插件相關(guān)的非局部變量,不能定義在類中,如果想全局調(diào)用只能放到Dictionary中,
  • 在調(diào)用插件卸載之前,刪除相關(guān)鍵值.
  • 任何通過插件返回的變量,不能為插件內(nèi)定義的變量類型.盡量使用json傳遞參數(shù).
  • 插件入口函數(shù)盡量使用同步函數(shù),如果為異步函數(shù),只能使用Task.Run方式裹所有邏輯.
  • 如果有任何疑問或不同意見,請賜教.

NFinal2開源框架。https://git.oschina.net/LucasDot/NFinal2/tree/master

到此這篇關(guān)于.Net core 的熱插拔機(jī)制的深入探索及卸載問題求救指南的文章就介紹到這了,更多相關(guān).Net core熱插拔機(jī)制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • .Net Core2.1 WebAPI新增Swagger插件詳解

標(biāo)簽:崇左 威海 伊春 銅川 撫州 西藏 晉城 深圳

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《.Net core 的熱插拔機(jī)制的深入探索及卸載問題求救指南》,本文關(guān)鍵詞  .Net,core,的,熱插,拔,機(jī)制,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《.Net core 的熱插拔機(jī)制的深入探索及卸載問題求救指南》相關(guān)的同類信息!
  • 本頁收集關(guān)于.Net core 的熱插拔機(jī)制的深入探索及卸載問題求救指南的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    美女精品自拍一二三四| 久久亚洲二区三区| 久久综合九色综合久久久精品综合| 久久蜜桃一区二区| 日日夜夜精品视频天天综合网| 91欧美一区二区| 欧美区在线观看| 亚洲精品成人在线| www.欧美日韩| 中国色在线观看另类| 国产成人无遮挡在线视频| 精品国产乱码久久| 国产高清无密码一区二区三区| 亚洲精品在线观看网站| 一区二区三区四区视频精品免费 | 国产高清不卡二三区| 日韩三级免费观看| 国产一区91精品张津瑜| 欧美人动与zoxxxx乱| 日韩在线观看一区二区| 欧美一级久久久久久久大片| 精品一区二区三区在线播放视频| 国产精品天美传媒沈樵| 一本色道亚洲精品aⅴ| 亚洲国产欧美在线人成| 国产欧美日产一区| 日韩精品一区二区在线观看| 99久精品国产| 国产河南妇女毛片精品久久久| 一区二区三区影院| 国产女人18毛片水真多成人如厕| 在线观看免费视频综合| 国产成人精品1024| 美女视频黄 久久| 亚洲福利电影网| 亚洲女人的天堂| 国产欧美精品在线观看| 精品国产污污免费网站入口 | 国产区在线观看成人精品| 色噜噜狠狠成人中文综合| 国产精品综合av一区二区国产馆| 亚洲成人精品在线观看| 国产精品久久久久久妇女6080| 日韩免费观看高清完整版| 欧美日韩综合在线免费观看| 99久久99久久久精品齐齐| 国产精品一区二区三区四区| 免费在线观看日韩欧美| 男女男精品视频网| 久久不见久久见免费视频1| 麻豆精品一区二区| 日韩经典中文字幕一区| 日韩影院在线观看| 日日摸夜夜添夜夜添亚洲女人| 亚洲一级二级三级在线免费观看| 亚洲欧美另类小说| 一级女性全黄久久生活片免费| 一区二区三区在线观看网站| 亚洲一区在线视频| 视频一区中文字幕国产| 美女诱惑一区二区| 国产一区 二区| a4yy欧美一区二区三区| 在线观看日韩毛片| 欧美一级在线观看| 国产日韩欧美不卡| 亚洲欧美激情视频在线观看一区二区三区| 亚洲免费av高清| 亚洲人成小说网站色在线| 亚洲图片欧美综合| 另类小说图片综合网| 国产一区二区三区在线看麻豆| 国产成人丝袜美腿| 在线观看国产精品网站| 日韩一级高清毛片| 欧美激情在线观看视频免费| 中文字幕一区二区三区蜜月| 亚洲综合丝袜美腿| 另类小说图片综合网| aaa亚洲精品一二三区| 欧美三级蜜桃2在线观看| 精品国产凹凸成av人导航| 欧美国产禁国产网站cc| 亚洲一区二区三区四区在线免费观看| 亚洲.国产.中文慕字在线| 激情综合色综合久久综合| 成人黄色网址在线观看| 欧美私模裸体表演在线观看| 精品久久久久久最新网址| 国产精品美女久久福利网站 | 青青草原综合久久大伊人精品优势| 精品一区二区三区视频在线观看 | 精品久久久久久久久久久久久久久久久| 久久精品亚洲精品国产欧美| 亚洲你懂的在线视频| 蜜臀av一区二区在线免费观看| 97se狠狠狠综合亚洲狠狠| 欧美群妇大交群中文字幕| 亚洲国产精品精华液2区45| 亚洲网友自拍偷拍| eeuss国产一区二区三区| 日韩三级精品电影久久久| 亚洲精品国产成人久久av盗摄 | www.亚洲免费av| 欧美日韩日日摸| 亚洲欧洲日本在线| 国内精品国产三级国产a久久| 欧洲亚洲国产日韩| 国产精品天干天干在观线| 精品一区二区免费| 精品视频999| 伊人夜夜躁av伊人久久| 高清av一区二区| 亚洲精品在线观看网站| 日韩av网站免费在线| 91蜜桃在线免费视频| 国产欧美日韩卡一| 国产精品888| 欧美精品一区视频| 免费观看在线色综合| 欧美伦理视频网站| 亚洲一区二区精品久久av| 一本色道亚洲精品aⅴ| 国产精品久久影院| 国产91精品免费| 国产精品人成在线观看免费 | 久久99精品国产.久久久久久| 精品视频一区二区三区免费| 亚洲综合丁香婷婷六月香| 欧洲av一区二区嗯嗯嗯啊| 亚洲欧美一区二区久久 | 中文字幕一区二区视频| 91在线观看美女| 中文字幕一区二区三区不卡在线| 99视频超级精品| 亚洲综合自拍偷拍| 欧美日韩mp4| 蜜桃一区二区三区在线| 久久天天做天天爱综合色| 床上的激情91.| 亚洲欧洲99久久| 欧美老人xxxx18| 激情综合色播激情啊| 久久久久久久久蜜桃| 国产mv日韩mv欧美| 亚洲视频香蕉人妖| 欧美无砖专区一中文字| 日韩黄色免费电影| 久久久亚洲高清| 99久久精品久久久久久清纯| 亚洲综合一区在线| 日韩一二三区不卡| 不卡电影一区二区三区| 亚洲精品视频一区二区| 欧美美女一区二区在线观看| 久久成人av少妇免费| 国产精品久久久久久妇女6080 | 91老师国产黑色丝袜在线| 午夜电影一区二区| 国产区在线观看成人精品| 欧美熟乱第一页| 国产尤物一区二区在线| 一区二区三区四区在线播放| 日韩午夜激情电影| 99久久国产综合精品色伊| 欧美aaaaa成人免费观看视频| 欧美极品aⅴ影院| 777午夜精品视频在线播放| 国产成人免费在线| 午夜激情一区二区三区| 国产调教视频一区| 91精品国产福利| 不卡av在线网| 久久国产尿小便嘘嘘| 亚洲综合小说图片| 国产精品天美传媒| 久久久久久久久免费| 8v天堂国产在线一区二区| 91香蕉国产在线观看软件| 经典三级视频一区| 视频一区视频二区中文字幕| 成人免费在线观看入口| 久久蜜桃av一区精品变态类天堂 | 午夜欧美在线一二页| 欧美激情一区二区三区| 日韩视频一区二区| 欧美精品一二三| 日本精品免费观看高清观看| 国产福利电影一区二区三区| 美女脱光内衣内裤视频久久网站| 亚洲www啪成人一区二区麻豆| 亚洲人快播电影网| 亚洲男人的天堂一区二区| 国产精品免费网站在线观看| 精品久久久久久无| 久久综合九色综合欧美亚洲| 欧美一区二区久久| 欧美精品亚洲二区| 日韩视频免费观看高清完整版在线观看 | 婷婷综合久久一区二区三区|