C#将dll打包到程序中的具体实现

时间:2022-11-22 16:39:34 

直接进入主题

先来看一个栗子,假设现在有一个第三方dll


namespace TestLibrary1
{
    public class Test
    {
        public void Point()
        {
            Console.WriteLine("aaabbbccc");
        }
    }
}


TestLibrary1.dll

在项目中引用,然后调用其中的方法Test,将输出aaabbbccc


using System;

namespace ConsoleApplication5
{
    class Program
    {
        static void Main(string[] args)
        {
            var test = new TestLibrary1.Test();
            test.Point();
            Console.ReadLine();
        }
    }
}

效果

C#将dll打包到程序中的具体实现

但是很显然,当你把程序发给你的客户的时候必须要携带一个dll,否则就会这样

C#将dll打包到程序中的具体实现

当程序在运行中,某个程序集加载失败的时候 会触发  AppDomain.CurrentDomain.AssemblyResolve 事件


//
// 摘要:
//     在对程序集的解析失败时发生。
public event ResolveEventHandler AssemblyResolve;


在这个事件中,可以重新为加载失败的程序集手动加载

如果你将dll作为资源文件打包的你的应用程序中(或者类库中)

C#将dll打包到程序中的具体实现

就可以在硬盘加载失败的时候 从资源文件中加载对应的dll

就像这样:


class Program
{
    static Program()
    {
        //这个绑定事件必须要在引用到TestLibrary1这个程序集的方法之前,注意是方法之前,不是语句之间,就算语句是在方法最后一行,在进入方法的时候就会加载程序集,如果这个时候没有绑定事件,则直接抛出异常,或者程序终止了
        AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;
    }

    static System.Reflection.Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
    {
        //获取加载失败的程序集的全名
        var assName = new AssemblyName(args.Name).FullName;
        if (args.Name == "TestLibrary1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null")
        {
            //读取资源
            using (var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream("ConsoleApplication5.TestLibrary1.dll"))
            {
                var bytes = new byte[stream.Length];
                stream.Read(bytes, 0, (int)stream.Length);
                return Assembly.Load(bytes);//加载资源文件中的dll,代替加载失败的程序集
            }
        }
        throw new DllNotFoundException(assName);
    }
    //程序进入方法之前会加载程序集,当程序集加载失败,则会进入CurrentDomain_AssemblyResolve事件
    static void Main(string[] args)
    {
        var test = new TestLibrary1.Test();
        test.Point();
        Console.ReadLine();
    }
}

这样就软件以一个exe单独运行了

C#将dll打包到程序中的具体实现

以上都是我网上看来了...................

--------------------------------------------------------------------------------
不过如果我有很多dll怎么办,总不至于每一个dll写一个分支吧?

所以我准备写一个通用的资源dll加载类

C#将dll打包到程序中的具体实现

原理蛮简单的,主要是通过StackTrace类获取调用RegistDLL方法的对象,获取到对方的程序集

然后通过Assembly.GetManifestResourceNames()获取所有资源的名称

判断后缀名".dll"(这一步可以自由发挥),然后加载,以加载的程序集的名称为key保存到一个字典中

并绑定AppDomain.AssemblyResolve事件

在程序集加载失败时,从字典中查询同名程序集,如果有,直接从字典中加载

代码如下:


using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Reflection;

namespace blqw
{
    /// <summary> 载入资源中的动态链接库(dll)文件
    /// </summary>
    static class LoadResourceDll
    {
        static Dictionary<string, Assembly> Dlls = new Dictionary<string, Assembly>();
        static Dictionary<string, object> Assemblies = new Dictionary<string, object>();

        static Assembly AssemblyResolve(object sender, ResolveEventArgs args)
        {
            //程序集
            Assembly ass;
            //获取加载失败的程序集的全名
            var assName = new AssemblyName(args.Name).FullName;
            //判断Dlls集合中是否有已加载的同名程序集
            if (Dlls.TryGetValue(assName, out ass) && ass != null)
            {
                Dlls[assName] = null;//如果有则置空并返回
                return ass;
            }
            else
            {
                throw new DllNotFoundException(assName);//否则抛出加载失败的异常
            }
        }

        /// <summary> 注册资源中的dll
        /// </summary>
        public static void RegistDLL()
        {
            //获取调用者的程序集
            var ass = new StackTrace(0).GetFrame(1).GetMethod().Module.Assembly;
            //判断程序集是否已经处理
            if (Assemblies.ContainsKey(ass.FullName))
            {
                return;
            }
            //程序集加入已处理集合
            Assemblies.Add(ass.FullName, null);
            //绑定程序集加载失败事件(这里我测试了,就算重复绑也是没关系的)
            AppDomain.CurrentDomain.AssemblyResolve += AssemblyResolve;
            //获取所有资源文件文件名
            var res = ass.GetManifestResourceNames();
            foreach (var r in res)
            {
                //如果是dll,则加载
                if (r.EndsWith(".dll", StringComparison.OrdinalIgnoreCase))
                {
                    try
                    {
                        var s = ass.GetManifestResourceStream(r);
                        var bts = new byte[s.Length];
                        s.Read(bts, 0, (int)s.Length);
                        var da = Assembly.Load(bts);
                        //判断是否已经加载
                        if (Dlls.ContainsKey(da.FullName))
                        {
                            continue;
                        }
                        Dlls[da.FullName] = da;
                    }
                    catch
                    {
                        //加载失败就算了...
                    }
                }
            }
        }
    }
}


LoadResource.Dll

标签:C#,dll,打包到程序
0
投稿

猜你喜欢

  • Java利用Selenium操作浏览器的示例详解

    2022-06-17 17:34:20
  • Java中实现简单的Excel导出

    2021-10-21 07:23:25
  • Android如何利用RecyclerView实现列表倒计时效果实例代码

    2023-01-24 08:08:53
  • Java8之lambda最佳实践_动力节点Java学院整理

    2023-11-28 00:07:28
  • Android7.0中关于ContentProvider组件详解

    2023-10-30 19:48:29
  • 浅谈利用Spring的AbstractRoutingDataSource解决多数据源的问题

    2021-09-07 07:20:59
  • 探讨Object转为String的几种简易形式详解

    2023-08-21 11:47:27
  • Java 方法(方法的定义,可变参数,参数的传递问题,方法重载,方法签名)

    2021-07-29 20:17:19
  • Java中transient关键字的详细总结

    2021-12-11 03:19:33
  • Flutter自定义底部导航栏的方法

    2022-01-10 15:48:33
  • Java求两集合中元素交集的四种方法对比分析

    2023-08-23 09:24:56
  • Android短信接收监听、自动回复短信操作例子

    2022-12-04 20:07:50
  • 浅谈c++11线程的互斥量

    2023-02-14 18:00:44
  • C# 迭代器分部类与索引器详情

    2023-10-16 21:18:18
  • c++回调之利用函数指针示例

    2022-07-26 06:59:32
  • Spring Cloud Gateway不同频率限流的解决方案(每分钟,每小时,每天)

    2023-01-05 13:49:34
  • java模拟实现银行ATM机操作

    2021-09-03 04:28:11
  • Android 实现左滑出现删除选项

    2021-05-28 12:05:59
  • 在Android设备上搭建Web服务器的方法

    2023-06-23 23:38:36
  • Springboot自带定时任务实现动态配置Cron参数方式

    2023-11-10 10:21:31
  • asp之家 软件编程 m.aspxhome.com