xuzeyu
xuzeyu
发布于 2024-08-26 / 80 阅读
0

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

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

新版本简介

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

核心编程揭秘

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


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

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

var loadContext = new AssemblyLoadContext("AntSKLoadContext", true);
public void 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);
                }
                return null;
            };
            // 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 (var type in 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 (var type in 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的插件化魔法,公众号下期将提供详细代码教程及其运用案例,敬请期待!别让知识止步于此,让我们在共享知识的海洋中,尽情航行吧!

相关文章

从 AntSK 接入讯飞星火大模型,看如何为 Semantic Kernel 对接无OpenAI接口协议的大模型平台

想利用大模型驱动真实业务系统?基于Semantic Kernel 接入讯飞星火 Function Call 调用本地方法!

深入解析:AntSK 0.1.7版本的技术革新与多模型管理策略

智能升级:AntSK教你如何让聊天机器人实现智能联网操作

AntSK:打造你的本地AI知识库——离线运行详细教程

基于AntSK与LLamaSharp打造私人定制的离线AI知识库