当前位置: 欣欣网 > 码农

AntSK 0.2.1 版本揭秘:动态加载dll,驱动Function Call新境界!

2024-03-17码农

在.NET的无限宇宙中,动态加载dll似乎一直是操控代码生生不息的魔杖。今天,我将与您探讨如何通过AntSK 0.2.1 版本灵活运用dll,将Function Call的强大功能插拔自如地融入项目之中,我们走入插件化开发的全新篇章。

新版本简介

AntSK,这个曾被我们广泛探讨过的Semantic Kernel项目,这次再度带来惊喜。在早先版本我们已经见识了其实现Function Call的威力。今日,我们有幸见证其升级再进化,0.2.1 版本究竟有何过人之处?答案就是:「动态加载dll」,这一核心玄机助您将代码实现插件化,不再拘泥于单一的AntSK平台编写,带来更加灵活的扩展性。

核心编程揭秘

要想细致品味动态加载dll的精妙,首先要解开的锁就是「约定」。AntSK要求插件明晰其身份,一段标明「AntSK」特性的代码如同通行证,使得函数得以插入AntSK的世界。

public classTestFunctionImport{/// <summary>/// 获取名称/// </summary>/// <returns>返回名称</returns> [Description("AntSK")]public string GetName() {return $""" 我的名字是AntSK, 我的作者是许泽宇 我是一个AI 知识库/智能体项目 """; }}

接下来,动态加载这块宝藏怎能轻易让人得逞?我们需要通过 AssemblyLoadContext 这位严厉的守门人,创建一个隔离的上下文,一步步引领我们:

var loadContext = new AssemblyLoadContext("AntSKLoadContext", true);publicvoid FuncLoad(string pluginPath){try {if (File.Exists(pluginPath)) {string directory = Path.GetDirectoryName(pluginPath);string fileName = Path.GetFileName(pluginPath);var resolver = new AssemblyDependencyResolver(directory);// Create a custom AssemblyLoadContext loadContext.Resolving += (context, assemblyName) => {string assemblyPath = resolver.ResolveAssemblyToPath(assemblyName);if (assemblyPath != null) {return context.LoadFromAssemblyPath(assemblyPath); }returnnull; };// Load your assembly Assembly pluginAssembly = loadContext.LoadFromAssemblyPath(pluginPath); } }catch (Exception ex) { Console.WriteLine(ex.Message + " ---- " + ex.StackTrace); }}

我们是如何评判「AntSK插件」芳容?以下搜寻被特性标记的方法:

publicvoid SearchMarkedMethods(){var markedMethods = new List<MethodInfo>(); _methodCache.Clear(); _methodInfos.Clear(); foreach (var assembly in _assemblies) {// 从缓存中获取标记了ActionAttribute的方法 foreach (vartypein assembly.GetTypes()) { markedMethods.AddRange(type.GetMethods().Where(m => { DescriptionAttribute da = (DescriptionAttribute)m.GetCustomAttributes(typeof(DescriptionAttribute), true).FirstOrDefault();return da != null && da.Description == "AntSK"; })); } }//动态加载部分var loadedAssemblies = loadContext.Assemblies.ToList(); foreach (var assembly in loadedAssemblies) {// 从缓存中获取标记了ActionAttribute的方法 foreach (vartypein assembly.GetTypes()) { markedMethods.AddRange(type.GetMethods().Where(m => { DescriptionAttribute da = (DescriptionAttribute)m.GetCustomAttributes(typeof(DescriptionAttribute), true).FirstOrDefault();return da != null && da.Description == "AntSK"; })); } }// 构建方法调用 foreach (var method in markedMethods) {var key = $"{method.DeclaringType.Assembly.GetName().Name}_{method.DeclaringType.Name}_{method.Name}";string pattern = "[^a-zA-Z0-9_]";// 使用 '-' 替换非ASCII的正则表达式的字符 key = Regex.Replace(key, pattern, "_"); _methodCache.TryAdd(key, method);var xmlCommentHelper = new XmlCommentHelper(); xmlCommentHelper.LoadAll();var description = xmlCommentHelper.GetMethodComment(method);var dict = xmlCommentHelper.GetParameterComments(method);var parameters = method.GetParameters().Select(x => (x.Name, x.ParameterType, dict[x.Name])).ToArray();var returnType = xmlCommentHelper.GetMethodReturnComment(method);if (string.IsNullOrEmpty(description)) { description = "导入插件"; } _methodInfos.TryAdd(key, (description, (method.ReflectedType, returnType), parameters)); }}

技术深度解读

动态加载dll的涵义远不止于表面的灵活,它开辟了无需重启应用程序即可更新程序功能的可能。通过以上技巧,我们能够在代码运行时插入或移除功能模块,极大地提升了代码的模块化和可维护性。

结语

搭上.Net技术的快车,我们仿佛有了横穿时空的能力。AntSK 0.2.1版本正是这趟快车上一颗灿烂的星子,动态加载,插件化编程,让Function Call这一古老而又强大的术语,获得了新的生命力。而今日的揭秘之旅,不知是否已让您心潮澎湃,跃跃欲试?

别忘了,每一段代码都饱含着程序员的智慧与汗水,它们值得我们去细细玩味和传唱。如果想要了解更多.Net技术,别忘了关注我的公众号,后续还有更多精彩内容等待着你来探索。向着代码的化简难度,以及编程模块化的美好未来,我们携手同行,共同进步!

福利补充

为了让您更好地理解AntSK的插件化魔法,公众号下期将提供详细代码教程及其运用案例,敬请期待!别让知识止步于此,让我们在共享知识的海洋中,尽情航行吧!

相关文章