成人性生交大片免费看视频r_亚洲综合极品香蕉久久网_在线视频免费观看一区_亚洲精品亚洲人成人网在线播放_国产精品毛片av_久久久久国产精品www_亚洲国产一区二区三区在线播_日韩一区二区三区四区区区_亚洲精品国产无套在线观_国产免费www

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

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

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

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

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

{
 "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())
 {
 //得到對應的實體文件
 var dependencyContext = 
 dependencyContextJsonReader.Read(dependencyFileStream);
 //定義的運行環(huán)境,沒有,則為全平臺運行.
 string currentRuntimeIdentifier= dependencyContext.Target.Runtime;
 //運行時所需要的dll文件
 var assemblyNames= dependencyContext.RuntimeLibraries;
 }
}

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

安裝 Microsoft.NETCore.Platforms包,并找到runtime.json運行時定義文件.

{
 "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依賴關系示意圖

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

.Net core常用發(fā)布平臺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é)點下,定義了所有的RID字典表以及RID樹關系.

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

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

三.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文件,并調用Load和LoadUnmanagedDll函數(shù)
 _resolver = new AssemblyDependencyResolver(pluginFolder);
 //第6步,通過第4,5步,解析仍失敗的dll會自動嘗試調用主程序中的程序集,
 //如果失敗,則直接拋出程序集無法加載的錯誤
 }
 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加載邏輯
 }
}

微軟官方示例代碼如下:示例具體內容

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,則直接加載主項目程序集
 return null;
 }

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

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

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

3. 手動加載時可以根據(jù)deps.json文件定義的runtime加載當前平臺下的unmanaged dll文件.

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

四.插件項目一定要和主項目使用同樣的運行時.

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

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

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

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

插件

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

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

3. 使用WeakReference關聯(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就會變?yōu)镕alse,即AssemblyLoadContext卸載失敗.
 for (int i = 0; weakReference.IsAlive  (i  10); i++)
 {
 GC.Collect();
 GC.WaitForPendingFinalizers();
 }
 }

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

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

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. 以上測試可以看出,如果插件調用的是一個常規(guī)帶wait的async異步函數(shù),則插件一定會卸載失敗.
原因推測是返回的結果是編譯器自動生成的狀態(tài)機實現(xiàn)的,而狀態(tài)機是在插件中定義的.

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

八.正確卸載插件的方式

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

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

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

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

標簽:崇左 威海 伊春 銅川 撫州 西藏 晉城 深圳

巨人網(wǎng)絡通訊聲明:本文標題《.Net core 的熱插拔機制的深入探索及卸載問題求救指南》,本文關鍵詞  .Net,core,的,熱插,拔,機制,;如發(fā)現(xiàn)本文內容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《.Net core 的熱插拔機制的深入探索及卸載問題求救指南》相關的同類信息!
  • 本頁收集關于.Net core 的熱插拔機制的深入探索及卸載問題求救指南的相關信息資訊供網(wǎng)民參考!
  • 推薦文章
    欧美黄色片在线观看| 久久人人爽人人爽人人av| 国产91精品一区二区麻豆网站| 欧美精品99久久久| wwww亚洲| 亚洲精品成人精品456| 精品久久久香蕉免费精品视频| 免费亚洲电影在线| 国产精品高清亚洲| 成人精品高清在线视频| 欧美亚洲免费电影| 欧美专区福利免费| 欧美成人乱码一区二区三区| 韩剧1988免费观看全集| 国产在线播放91| 91精品国模一区二区三区| 日本激情一区二区三区| 欧美日韩中文在线| 在线一区二区三区四区五区| jizz亚洲大全| 亚洲一区二区三区美女| 91精品国产91久久久久久青草| 午夜亚洲成人| 国产麻豆电影在线观看| 亚洲国产精品久久一线不卡| 亚洲精品电影网| 青青久久av北条麻妃海外网| 国产成人精品综合在线观看| 欧美在线观看成人| 亚洲天天综合| av在线不卡精品| 免费av一区| 欧美18视频| 亚洲色图制服诱惑| 成人午夜大片| 黄色录像一级片| 国产精品人人爽| 91在线观看喷潮| 麻豆91精品视频| 国产999在线观看| 一级黄色片网站| 美女被久久久| 国产女人18毛片水18精| 国产精品二区一区二区aⅴ污介绍| 九九热国产精品视频| 亚洲色图综合图区| 欧美牲交a欧美牲交aⅴ免费下载| 国产精品美女久久久| 日韩免费一区二区三区| 亚洲av永久无码国产精品久久| www.超碰com| av在线不卡网| 亚洲做受高潮无遮挡| 视频精品在线观看| 久激情内射婷内射蜜桃| 精品一区二区成人精品| 亚洲成人网av| china中国猛gary| 电影一区二区三区| 久久99热国产| 色婷婷成人综合| 精品免费99久久| 国产日本欧美在线观看| 波多野结衣高清在线| 青青草手机视频在线观看| 一区二区三区国产精品| 一级片一区二区三区| 国产羞羞视频在线播放| 少妇愉情理伦片bd| 亚洲午夜伦理| 国产精品久久免费看| 97超碰在线公开在线看免费| 一本久道中文无码字幕av| 精品全国在线一区二区| 男女啪啪免费观看| 在线播放国产视频| 欧美黑人一区二区三区| 国产一区视频在线看| 免费视频最近日韩| 三级精品在线观看| 激情小说网站亚洲综合网| 日韩va亚洲va欧洲va国产| 亚洲欧美久久婷婷爱综合一区天堂| 久久国产精品亚洲77777| 最大av网站| 正在播放一区二区三区| 影音四色91| 亚洲精品久久一区二区三区777| 99九九热只有国产精品| 国产精品最新乱视频二区| 国产精品一区二区三| 91超碰免费在线| 欧美大香线蕉线伊人久久| 日韩精品久久久免费观看| a天堂中文在线观看| 人妻熟女一二三区夜夜爱| 国产91在线播放九色| 精品午夜久久福利影院| 国产成人精品男人的天堂538| 杨幂一区欧美专区| 性中国古装videossex| 26uuu久久天堂性欧美| 日韩中文字幕视频在线观看| 91免费视频网址| 激情成人综合| 制服丝袜在线第一页| 尤物视频网站在线观看| 一区二区三区不卡视频在线观看| av在线最新| 亚洲天堂成人在线视频| 极品国产91在线网站| aaa免费看大片| 亚州国产精品久久久| 中文字幕一区二区三区精彩视频| 伊人久久亚洲综合| 欧美在线一级片| 2020国产在线视频| 国产成人午夜精品影院观看视频| 国产精品极品美女粉嫩高清在线| 91嫩草亚洲精品| 91精品蜜臀在线一区尤物| 成人线上播放| 午夜亚洲国产au精品一区二区| 午夜精品一区二区三区在线视| 欧美日韩一区三区四区| 国产精品久久久久久久电影| 国产精品美乳一区二区免费| 中文无码久久精品| 精品国产一区久久久| 日本在线影院| 性欧美xxxx视频在线观看| 欧美精品在欧美一区二区| 欧美一区二区三区在线电影| 欧美成人a∨高清免费观看| h在线观看视频免费网站| 国产农村妇女精品一二区| 国产精品乱人伦一区二区| 天天色综合社区| 久久久精品网站| www.国产一区二区| 超级白嫩亚洲国产第一| 欧美一级大片在线视频| 99久热在线精品996热是什么| 亚洲无限av看| 午夜精品久久久久久久91蜜桃| 91精品国产综合久久小美女| 国产精品亚洲аv天堂网| 国产精品mp4| 中文字幕av在线| 国产一区二区三区无遮挡| 久草福利资源在线| 免费在线观看av片| 亚洲精品国产成人av在线| 国产精品偷伦免费视频观看的| 日韩中文字幕久久| 欧美一级小视频| 精品国产欧美一区二区| 欧洲中文字幕国产精品| 欧美下载看逼逼| 最近中文字幕免费在线观看| 色噜噜色狠狠狠狠狠综合色一| 日本视频精品一区| 国产一级做a爱片久久毛片a| 45www国产精品网站| 伊人成综合网伊人222| 亚洲第一第二区| 久久久久久久伊人| 狼人精品一区二区三区在线| 黄网av在线| 97视频免费在线观看| 日本三级韩国三级欧美三级| 影音日韩av| 97伦理在线四区| 亚洲欧美春色| 日韩在线一级片| 亚洲一级二级三级在线免费观看| 中文字幕在线有码| www日韩视频| 国产福利在线看| 国产精品福利在线观看网址| 综合色中文字幕| 女尊高h男高潮呻吟| 2023国产精品| 国产精品一区二区免费不卡| 日本免费一区视频| 久久久久久天堂| 欧美性片在线观看| 午夜精品影视国产一区在线麻豆| 国产成人免费看一级大黄| 一区三区视频| aa视频在线播放| 精品国产第一福利网站| 国产永久免费视频| 国产一区二区在线看| 亚洲欧美视频一区二区| 亚洲图片欧美日韩| 成人综合婷婷国产精品久久| 日韩精品久久理论片| 奇米影视首页 狠狠色丁香婷婷久久综合| 欧美午夜宅男影院在线观看| 九热视频在线观看| 欧美性猛交xxxxbbb| 北条麻妃av高潮尖叫在线观看| 国产综合色在线观看| 好吊视频一区二区三区| 欧美 日韩 激情| 国产自偷自偷免费一区| 亚洲乱码国产乱码精品精天堂| 免费观看一级特黄欧美大片| 国产真实乱对白精彩久久| 9久re热视频在线精品| 日韩黄色片网站| 欧美综合另类| 四虎影视精品永久在线观看| 91久久国产最好的精华液| 日本网站在线观看一区二区三区| 国产精品免费在线免费| 久久精品系列| 国产一区二区三区四区五区六区| 欧美影院在线| 色噜噜狠狠成人中文综合| 亚洲国产欧美一区二区丝袜黑人| bt欧美亚洲午夜电影天堂| 蜜臀av一级做a爰片久久| 日精品一区二区| 在线看女人毛片| 玛雅亚洲电影| 欧美精品日韩一区| 99视频在线观看地址| 免费在线超碰| a视频免费在线观看| 99精品免费观看| 男人的天堂va在线| 欧美午夜电影网| 日韩一级大片在线观看| 国产精品高潮视频| 老司机精品视频一区二区三区| 欧美日韩视频免费在线观看| 色天使色偷偷av一区二区| 亚洲午夜精品一区二区三区| 男人添女人荫蒂免费视频| 久久免费公开视频| 美女国内精品自产拍在线播放| 精油按摩中文字幕久久| 国产女主播一区二区| 欧美日韩123| 大伊香蕉精品在线品播放| 国产成人三级在线观看| 久久久久久五月天久久久久久久久| 日韩片之四级片| 国产免费一区二区三区视频| 黄色录像a级片| 精品国产第一区二区三区观看体验| 亚洲男人的天堂一区二区| 99久久精品费精品国产一区二区| tube8在线hd| 成人3d动漫一区二区三区91| 男人天堂影院| 黄页大全在线免费观看| 久久久91精品国产一区不卡| 美女福利一区二区三区| av电影在线网站| 真实的国产乱xxxx在线91| jizz性欧美23| 在线免费亚洲电影| 日本久久二区| 久久久国产视频| 国语对白精品一区二区| 中文字幕在线看片| 久久综合另类图片小说| 日本福利片高清在线观看| 久久久久国产一区二区三区四区| 亚洲午夜精品一区 二区 三区| 国内自拍亚洲| 国产视频中文字幕在线观看| 久热成人在线视频| 天堂…中文在线最新版在线| 日韩第一页在线观看| 国产精品冒白浆免费视频| 欧美高清成人| 精品福利视频导航大全| 日韩偷拍自拍| 国v精品久久久网| 97国产一区二区| 69**夜色精品国产69乱| 国产精自产拍久久久久久蜜| 免费精品国产| a√资源在线| 精品香蕉在线观看视频一| 一区二区三区中文在线观看| 韩国一区二区在线播放| 一区二区在线观看不卡| 黄色仓库视频网站| 欧美午夜性生活| 日韩欧美2区| 黄色免费在线播放| 日韩国产一区三区| 亚洲视频在线观看一区| 欧美性生交xxxxx久久久| 日本激情在线观看| 日本三级视频在线| 欧美黑人视频一区| 久久精品99久久久| 国产亚洲精品成人| 91精品观看| 欧美1o一11sex性hdhd| 亚洲综合视频网| 国产羞羞视频在线播放| 成年人的黄色片| 亚洲永久精品一区| 中文字幕在线观看第二页| 午夜成人在线视频| 人妻有码中文字幕| 亚洲另类黄色| 国产亚洲精品久久777777| 亚洲涩涩在线观看| 日韩精品中文字幕第1页| 亚洲成人免费看| 日韩经典在线视频| 在线看av网址| 欧美a一欧美| va天堂va亚洲va影视| 亚洲国产精品成人va在线观看| 93在线视频精品免费观看| 久久国产精品波多野结衣| 97精品国产| 色屁屁草草影院ccyycom| 六月丁香婷婷综合|