.net加载失败的程序集重新加载

在.net程序中,程序集是Lazy加载的,只有在用的时候才会去加载,当程序集加载失败时,会触发AppDomain.AssemblyResolve的事件,在这个事件中,我们甚至还可以进行补救,从别得地方重新加载程序集。

AppDomain.CurrentDomain.AssemblyResolve += (s, e) =>
{
    byte[] content = getLibBytes(e.Name);
    return Assembly.Load(content);
};

这个功能如果使用起来就非常灵活了,它可以控制我们自由控制程序集的加载方式。常用的方法有如下几个:

 

程序集保护:

.net程序是非常容易反编译的,这个特性提供了混淆外的另一个方式。由于动态调用的方式下,程序集不需要是原始dll,甚至都不需要存储在磁盘上。可以通过直接不让使用者获取到程序集的dll的方式防止反编译。

发布的程序的时候,不直接发布需要保护的程序集,将程序集加密后发布,或者直接加密后存储在服务器上。使用的时候,在AssemblyResolve中获取加密后的程序集,解密后返回。

 

程序集合并:

WPF程序由于使用了反射,使用传统的ILMerge的方式合并后,由于程序集变化了,往往不能正常工作。

有很多工具,通过将程序集合并到exe的资源文件中,使用的时候,再在ssemblyResolve中从资源文件中获取程序集返回。具体可以参看我之前的文章: 使用LibZ合并.Net程序集

 

客户端更新:

CS模式的程序一个不足就是更新不方便,可以将程序集存储在文件数据库中,直接更新程序集数据库就可以很方便的实现程序集更新。

 

程序集存储分离:

使用微服务模式时,很多部署在同一个服务器上的服务共用着相同的程序集(第三方的Nuget库),这些程序集更新频率很低,并且混在一起存储使得我们不容易找到业务程序集。

可以将这些程序集集中存储在独立的位置。服务文件夹中只发布我们的业务程序集,看起来更加清晰,更新也更加方便。

 

.net core

在.net core中,这个机制也是可以使用的,不过接口发生了一点变化:

AssemblyLoadContext.Default.Resolving += (context, assembly) =>
{
    var content = getLibBytes(assembly.FullName);
    return Assembly.Load(content);
};

需要说明的是,如果是使用 dotnet xxx.dll 的方式运行的话,dotnet 程序会首先通过 xxx.deps.json文件来获取所有相关的依赖性,从而还没有进入程序就报错。

可以通过修改 xxx.deps.json去掉依赖项,或者干脆直接删掉xxx.deps.json解决这个问题。

   

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您好!感谢您的提问。如果您想从程序加载自定义的邮箱(Mailbox),可以按照以下步骤进行操作: 1. 创建自定义邮箱类,继承自 MailboxType。 ``` public class MyMailbox : MailboxType { public MyMailbox(Settings settings, Config config) : base(settings, config) { // 可以在这里进行一些初始化操作 } // 创建邮箱实例 public override IMessageQueue Create(IActorRef owner, ActorSystem system) { // 返回自定义的邮箱实例 return new MyMessageQueue(); } } ``` 2. 创建自定义消息队列类,实现 IMessageQueue 接口。 ``` public class MyMessageQueue : IMessageQueue { // 实现 IMessageQueue 接口中的方法 // ... } ``` 3. 在程序中注册自定义邮箱。 ``` // 注册邮箱 var mailboxConfig = ConfigurationFactory.ParseString(@" akka.actor.mailbox.requirements { ""my-mailbox"" = ""MyNamespace.MyMailbox, MyAssemblyName"" } "); // 加载程序 var assembly = Assembly.LoadFrom("MyAssemblyName.dll"); // 创建 ActorSystem var system = ActorSystem.Create("MySystem", ConfigurationFactory.Load(mailboxConfig).WithFallback(ConfigurationFactory.FromAssembly(assembly))); ``` 其中,`MyNamespace.MyMailbox` 是自定义邮箱类的完整命名空间和类名,`MyAssemblyName` 是包含自定义邮箱类的程序名称。 在创建 ActorSystem 时,使用 `ConfigurationFactory.Load(mailboxConfig).WithFallback(ConfigurationFactory.FromAssembly(assembly))` 方法加载配置信息。这样,程序就可以从程序加载自定义邮箱了。 4. 在创建 Actor 时,指定使用自定义邮箱。 ``` var props = Props.Create<MyActor>().WithMailbox("my-mailbox"); var actor = system.ActorOf(props, "MyActor"); ``` 在创建 Actor 时,使用 `WithMailbox` 方法指定使用自定义邮箱。 希望这些信息能够帮助到您!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值