基于.net standard 的動(dòng)態(tài)編譯實(shí)現(xiàn)代碼
在上篇文章[基于.net core 微服務(wù)的另類實(shí)現(xiàn)]結(jié)尾處,提到了如何方便自動(dòng)的生成微服務(wù)的客戶端代理,使對于調(diào)用方透明,同時(shí)將枯燥的東西使用框架集成,以提高使用便捷性。在嘗試了基于 Emit 中間語言后,最終決定使用生成代碼片段然后動(dòng)態(tài)編譯的模式實(shí)現(xiàn)。
1.背景:
其一在前文中,我們通過框架實(shí)現(xiàn)了微服務(wù)面向使用者的透明調(diào)用,但是需要為每個(gè)服務(wù)寫一個(gè)客戶端代理,顯得異常繁瑣,其二項(xiàng)目中前端站點(diǎn)使用了傳統(tǒng)的.Net Framework 框架,后端微服務(wù)我們使用了.Net Core 框架改造,短時(shí)間將前端站點(diǎn)調(diào)整成 .Net Core 框架亦不現(xiàn)實(shí),為了能同時(shí)支持這兩種框架。如何 .Net Standard 框架來自動(dòng)創(chuàng)建微服務(wù)的客戶端代理成為我們必須解決的問題。
2.問題轉(zhuǎn)化
我們在回頭簡單看一下我們現(xiàn)在期望的微服務(wù)客戶端代理長的樣子:

通過上面分析,我們只需要將服務(wù)接口中的每個(gè)方法,判斷是否有返回值,如果有返回值調(diào)用Invoke<ReturnType>方法,沒有返回值調(diào)用InvokeWithoutReturn方法,然后依次將接口名,方法名以及方法的參數(shù)按順序傳入即可。各位如果是熟悉Java的同學(xué),這個(gè)問題很容易解決,使用動(dòng)態(tài)代理創(chuàng)建一個(gè)這樣的匿名類即可,但在.net 的世界里,動(dòng)態(tài)代理的實(shí)現(xiàn)確顯得異常麻煩。
首先想到是通過中間語言 IL 的 Emit 實(shí)現(xiàn),但無奈這個(gè)使用起來實(shí)在是太不友好了, 幾經(jīng)折騰最終還是選擇放棄了,后又想到其實(shí)可以通過動(dòng)態(tài)生成這個(gè)代碼片段,動(dòng)態(tài)編譯后加載到系統(tǒng)程序集中,應(yīng)該就可以了。于是在這個(gè)方向的指引下,我們嘗試著去一步步實(shí)現(xiàn)這個(gè)問題。
3.解決方案
如何生成這個(gè)代碼片段? 通過上面的分析,我們知道只需要將接口反射獲取其中的公共方法,并將接口的每個(gè)方法簽名原樣復(fù)制,在根據(jù)接口方法是否有返回值分別調(diào)用RemoteServiceProxy基類中相關(guān)方法即可,不過需要特殊注意的泛型方法翻譯,以下是生成這個(gè)代碼片段的參考實(shí)現(xiàn).
尋找出為服務(wù)接口程序集文件,并處理每個(gè)文件
private static StringBuilder CreateApiProxyCode()
{
var path = GetBinPath();
var dir = new DirectoryInfo(path);
//獲取項(xiàng)目中微服務(wù)接口文件
var files = dir.GetFiles("XZL*.Api.dll");
var codeStringBuilder = new StringBuilder(1024);
//添加必要的using
codeStringBuilder
.AppendLine("using System;")
.AppendLine("using System.Collections.Generic;")
.AppendLine("using System.Text;")
.AppendLine("using XZL.Infrastructure.ApiService;")
.AppendLine("using XZL.Infrastructure.Defines;")
.AppendLine("using XZL.Model;")
.AppendLine("namespace XZL.ApiClientProxy")
.AppendLine("{"); //namespace begin
//處理每個(gè)文件中的接口信息
foreach (var file in files)
{
CreateApiProxyCodeFromFile(codeStringBuilder, file);
}
codeStringBuilder.AppendLine("}"); //namespace end
return codeStringBuilder;
}
處理每個(gè)文件中的接口類型,并將每個(gè)程序集的依賴程序集找出來,方便后面動(dòng)態(tài)編譯
private static void CreateApiProxyCodeFromFile(StringBuilder fileCodeBuilder, FileInfo file)
{
try
{
Assembly apiAssembly = Assembly.Load(file.Name.Substring(0, file.Name.Length - 4));
var types = apiAssembly
.GetTypes()
.Where(c => c.IsInterface && c.IsPublic)
.ToList();
var apiSvcType = typeof(IApiService);
bool isNeed = false;
foreach (Type type in types)
{
//找出期望的接口類型
if (!apiSvcType.IsAssignableFrom(type))
{
continue;
}
//找出接口的所有方法
var methods = type.GetMethods(BindingFlags.Public
| BindingFlags.FlattenHierarchy
| BindingFlags.Instance);
if (!methods.Any())
{
continue;
}
//定義代理類名,以及實(shí)現(xiàn)接口和繼承RemoteServiceProxy
fileCodeBuilder.AppendLine($"public class {type.FullName.Replace(".", "_")}Proxy :" +
$"RemoteServiceProxy, {type.FullName}")
.AppendLine("{"); //class begin
//處理每個(gè)方法
foreach (var mth in methods)
{
CreateApiProxyCodeFromMethod(fileCodeBuilder, type, mth);
}
fileCodeBuilder.AppendLine("}"); //class end
isNeed = true;
}
if (isNeed)
{
var apiRefAsms = apiAssembly.GetReferencedAssemblies();
refAssemblyList.Add(apiAssembly.GetName());
refAssemblyList.AddRange(apiRefAsms);
}
}
catch
{
}
}
處理接口中的每個(gè)方法
private static void CreateApiProxyCodeFromMethod(
StringBuilder fileCodeBuilder,
Type type,
MethodInfo mth)
{
var isMthReturn = !mth.ReturnType.Equals(typeof(void));
fileCodeBuilder.Append("public ");
//添加返回值
if (isMthReturn)
{
fileCodeBuilder.Append(GetFriendlyTypeName(mth.ReturnType)).Append(" ");
}
else
{
fileCodeBuilder.Append(" void ");
}
//方法參數(shù)開始
fileCodeBuilder.Append(mth.Name).Append("(");
var mthParams = mth.GetParameters();
if (mthParams.Any())
{
var mthparaList = new List<string>();
foreach (var p in mthParams)
{
mthparaList.Add(GetFriendlyTypeName(p.ParameterType) + " " + p.Name);
}
fileCodeBuilder.Append(string.Join(",", mthparaList));
}
//方法參數(shù)結(jié)束
fileCodeBuilder.Append(")");
//方法體開始
fileCodeBuilder.AppendLine("{");
if (isMthReturn)
{
//返回值
fileCodeBuilder.Append("return Invoke<")
.Append(GetFriendlyTypeName(mth.ReturnType))
.Append(">");
}
else
{
fileCodeBuilder.Append(" InvokeWithoutReturn");
}
//拼接接口名及方法名
fileCodeBuilder.Append($"(\"{type.FullName}\",\"{mth.Name}\"");
//方法本身參數(shù)
if (mthParams.Any())
{
fileCodeBuilder.Append(",").Append(string.Join(",", mthParams.Select(t => t.Name)));
}
fileCodeBuilder.Append(");");
//方法體結(jié)束
fileCodeBuilder.AppendLine("}");
}
獲取泛型類型字符串
private static string GetFriendlyTypeName(Type type)
{
if (!type.IsGenericType)
{
return type.FullName;
}
string friendlyName = type.Name;
int iBacktick = friendlyName.IndexOf('`');
if (iBacktick > 0)
{
friendlyName = friendlyName.Remove(iBacktick);
}
friendlyName += "<";
Type[] typeParameters = type.GetGenericArguments();
for (int i = 0; i < typeParameters.Length; ++i)
{
string typeParamName = GetFriendlyTypeName(typeParameters[i]);
friendlyName += (i == 0 ? typeParamName : "," + typeParamName);
}
friendlyName += ">";
return friendlyName;
}
如何添加依賴
既然是要編譯源碼,那么源碼中的依賴必不可少,在上一步中我們已經(jīng)將每個(gè)程序集的依賴一并找出,接下來我們將這些依賴全部整理出來
//緩存程序集依賴
var references = new List<MetadataReference>();
var refAsmFiles = new List<string>();
//系統(tǒng)依賴
var sysRefLocation = typeof(Enumerable).GetTypeInfo().Assembly.Location;
refAsmFiles.Add(sysRefLocation);
//refAsmFiles原本緩存的程序集依賴
refAsmFiles.Add(typeof(object).GetTypeInfo().Assembly.Location);
refAsmFiles.AddRange(refAssemblyList.Select(t => Assembly.Load(t).Location).Distinct().ToList());
//傳統(tǒng).NetFramework 需要添加mscorlib.dll
var coreDir = Directory.GetParent(sysRefLocation);
var mscorlibFile = coreDir.FullName + Path.DirectorySeparatorChar + "mscorlib.dll";
if (File.Exists(mscorlibFile))
{
references.Add(MetadataReference.CreateFromFile(mscorlibFile));
}
var apiAsms = refAsmFiles.Select(t => MetadataReference.CreateFromFile(t)).ToList();
references.AddRange(apiAsms);
//當(dāng)前程序集依賴
var thisAssembly = Assembly.GetEntryAssembly();
if (thisAssembly != null)
{
var referencedAssemblies = thisAssembly.GetReferencedAssemblies();
foreach (var referencedAssembly in referencedAssemblies)
{
var loadedAssembly = Assembly.Load(referencedAssembly);
references.Add(MetadataReference.CreateFromFile(loadedAssembly.Location));
}
}
編譯
有了代碼片段, 也有了編譯程序集依賴, 接下來就是最重要的編譯了.
//定義編譯后文件名
var path = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Proxy");
if (!Directory.Exists(path))
{
Directory.CreateDirectory(path);
}
var apiRemoteProxyDllFile = Path.Combine(path,
apiRemoteAsmName + DateTime.Now.ToString("yyyyMMddHHmmssfff") + ".dll");
var tree = SyntaxFactory.ParseSyntaxTree(codeBuilder.ToString());
var compilation = CSharpCompilation.Create(apiRemoteAsmName)
.WithOptions(new CSharpCompilationOptions(OutputKind.DynamicallyLinkedLibrary))
.AddReferences(references)
.AddSyntaxTrees(tree);
//執(zhí)行編譯
EmitResult compilationResult = compilation.Emit(apiRemoteProxyDllFile);
if (compilationResult.Success)
{
// Load the assembly
apiRemoteAsm = Assembly.LoadFrom(apiRemoteProxyDllFile);
}
else
{
foreach (Diagnostic codeIssue in compilationResult.Diagnostics)
{
string issue = $"ID: {codeIssue.Id}, Message: {codeIssue.GetMessage()}," +
$" Location: { codeIssue.Location.GetLineSpan()}, " +
$"Severity: { codeIssue.Severity}";
AppRuntimes.Instance.Loger.Error("自動(dòng)編譯代碼出現(xiàn)異常," + issue);
}
}
結(jié)語
在經(jīng)過以上處理后,雖算不上完美,但順利的實(shí)現(xiàn)了我們期望的樣子,在之前的GetService中,當(dāng)發(fā)現(xiàn)屬于遠(yuǎn)程服務(wù)的時(shí)候,只需要類似如下形式返回代理對象即可。同時(shí)為增加調(diào)用更加順暢,我們將此編譯的時(shí)機(jī)定在了發(fā)生在程序啟動(dòng)的時(shí)候,ps 當(dāng)然或許還有一些其他更合適的時(shí)機(jī).
static ConcurrentDictionary<string, Object> svcInstance = new ConcurrentDictionary<string, object>();
var typeName = "XZL.ApiClientProxy." + typeof(TService).FullName.Replace(".", "_") + "Proxy";
object obj = null;
if (svcInstance.TryGetValue(typeName, out obj) && obj != null)
{
return (TService)obj;
}
try
{
obj = (TService)apiRemoteAsm.CreateInstance(typeName);
svcInstance.TryAdd(typeName, obj);
}
catch
{
throw new ICVIPException($"未找到 {typeof(TService).FullName} 的有效代理");
}
return (TService)obj;
總結(jié)
以上所述是小編給大家介紹的基于.net standard 的動(dòng)態(tài)編譯實(shí)現(xiàn)代碼,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
- 基于Jenkins搭建.NET FrameWork持續(xù)集成環(huán)境
- 簡單了解.NET Framework
- .NET Core/Framework如何創(chuàng)建委托大幅度提高反射調(diào)用的性能詳解
- Windows Server 2012 R2或2016無法安裝.NET Framework 3.5.1的解決方法
- .NET framework 4.0 安裝失敗回滾問題
- 安裝.NET Framework進(jìn)度條卡住不動(dòng)的解決方案(推薦)
- Visual Studio 2017創(chuàng)建.net standard類庫編譯出錯(cuò)原因及解決方法
- .Net Framework .Net .NET Standard的概念及區(qū)別
相關(guān)文章
ASP.NET MVC中解析淘寶網(wǎng)頁出現(xiàn)亂碼問題的解決方法
最近在使用MVC解析淘寶網(wǎng)頁出現(xiàn)亂碼問題,原因就是中文字符格式出現(xiàn)沖突,ASP.NET MVC 默認(rèn)采用utf-8,但是淘寶網(wǎng)頁采用gbk。在網(wǎng)上找了一下,最常用的解決方法,特分享下2013-04-04
利用ASP.NET MVC和Bootstrap快速搭建個(gè)人博客之文章打賞功能(六)
這篇文章主要介紹了利用ASP.NET MVC和Bootstrap快速搭建個(gè)人博客之文章打賞功能(六) 的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07
ASP.NET Core中使用EPPlus導(dǎo)入出Excel文件的完整步驟
這篇文章主要給大家介紹了關(guān)于ASP.NET Core中如何使用EPPlus導(dǎo)入出Excel文件的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02
Asp.net簡單實(shí)現(xiàn)給圖片增加文字水印
這篇文章主要介紹了Asp.net簡單實(shí)現(xiàn)給圖片增加文字水印,需要的朋友可以參考下2014-12-12
asp.net IList查詢數(shù)據(jù)后格式化數(shù)據(jù)再綁定控件
這篇文章送給.net初學(xué)者或者遇到類似問題的朋友,就是IList如何格式化數(shù)據(jù)再綁定,我看到網(wǎng)上沒有多少朋友講到這方面的最基本的問題,現(xiàn)在我簡單說說吧,代碼我就截取其中一些講,如果不明白的朋友可以留言或者聯(lián)系我。2009-11-11
詳解ASP.NET-----Repeater數(shù)據(jù)控件的用法總結(jié)
本篇文章主要介紹了ASP.NET--Repeater數(shù)據(jù)控件的用法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。2016-11-11
ASP.NET 根據(jù)漢字獲取漢字拼音的首字母(含多音字)
本文分享了一個(gè)函數(shù),這個(gè)函數(shù)可以根據(jù)漢字的字符串獲取其拼音的首字母,以便我們在實(shí)際開發(fā)中使用。2016-04-04

