【转】C#开发奇技淫巧三:把dll放在不同的目录让你的程序更整洁

转自:https://www.cnblogs.com/marvin/p/PutDllToSpecificFolder.html?utm_source=tuicool&utm_medium=referral

 

系列文章

  • C#开发奇技淫巧一:调试windows系统服务
  • C#开发奇技淫巧二:根据dll文件加载C++或者Delphi插件
  • C#开发奇技淫巧三:把dll放在不同的目录让你的程序更整洁

程序目录的整理

想必C#的开发者都遇到过这个问题,引用的dll都放在根目录下,随着项目的日益增大,根目录下充满了各种各样的dll,非常的不美观。

如果能够把dll按照想要的目录来存放,那么系统就美观多了,以下是我常用的程序各文件的分布:

  • 【3rdLibs】
    • NLog.dll
    • Newtonsoft.Json.dll
    • ……
  • 【MyLibs】
  • 【Resources】
  • 【Images】
  • Excecutable.exe
  • Excecuteble.exe.config

网上有很多的文章述说这个,比如使用Assembly.Load,但是没有说明在程序中怎么使用,也没有给出具体的代码。这里我结合自己多年的实践经验,再把整个流程和方法详细叙述一遍,以便各位看官有个具体的体会。

系统搜索dll的目录以及顺序

CLR解析一个程序集会在一个根目录内进行搜索,整个探索过程又称Probing,这个根目录很显然就是当前包含当前程序集的目录。

AppDomainSetup这个类存储着探索目录的信息,其成员包括:ApplicationBasePrivateBinPath

程序搜索dll的顺序如下(区分强名称签名的和没有强名称签名的程序集):

没有做强名称签名的程序集:

  • 程序的根目录
  • 根目录下面,与被引用程序集同名的子目录
  • 根目录下面被明确定义为私有目录的子目录
  • 在目录中查找的时候,如果dll查找不到,则会尝试查找同名的exe
  • 如果程序集带有区域性,而不是语言中立的,则还会尝试查找以语言区域命名的子目录

具有强名称签名的程序集:

  • 全局程序集缓存
  • 如果有定义codebase,则以codebase定义为准,如果codebase指定的路径找不到,则直接报告错误
  • 程序的根目录
  • 根目录下面,与被引用程序集同名的子目录
  • 根目录下面被明确定义为私有目录的子目录
  • 在目录中查找的时候,如果dll查找不到,则会尝试查找同名的exe
  • 如果程序集带有区域性,而不是语言中立的,则还会尝试查找以语言区域命名的子目录

如何让程序识别不同目录下的dll?

我们看到,上面的顺序无论是否有强名称签名看,都提到了一个名词“私有目录”

方法一:配置App.config文件的privatePath——【推荐】

这是最简单的方法,当然也有一定的局限性,就是没法对dll做控制,另外,无法解决第三方DllImprt中引入的程序集不在根目录下的问题,不过无论怎么说,这个都基本解决了问题。

配置如下,多个目录用;分隔

<runtime><assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1"><probing privatePath="3rdLib;MyLibs;SubFolder\Sub.dll"/></assemblyBinding>
</runtime>

方法二:订阅程序集解析事件AssemblyResolve在代码中解析

应用程序集域中支持在程序集解析时的处理:AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve;。通过这个事件,我们可以在程序集解析时,根据不同的程序集做不用的处理,比如加载x86的程序集还是64位的程序集,当然也就可以指定程序集目录了

这也正是Assembly.LoadAssembly.LoadFrom等方法的用武之地。

Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args)
{AssemblyName assemblyName = new AssemblyName(args.Name);return Assembly.LoadFrom(Path.Combine(baseDirectory, "3rdLibs"));
}

方法三:在加载使用到dll的代码之前设置重置当前环境的目录

这个方法就是通过Environment.CurrentDirectory=customPath,这样,在调用dll方法时,因为目录已经切换到了
这是一个取巧的方法,不是很实用,要来回切换程序集目录,但是在某些情况下非常好用

如何处理[dllImport]中的程序集的加载

自己写dllImport

如果是自己写,那么久好控制了,可以直接指定相对的目录DllImport(3rdLibs\NLog.dll)。不过这种方法不一定可靠,在某些系统硬是加载不了,如果使用了dllImport还是,推荐下面的另外一种方法。

引用的C#的插件又使用了dllImport

这是很多文章都没有提及的:

因为无法更改路径,那么只能够使用上述特殊的方法,更改当前程序的路径

当然,还有更省事一点的做法,就是在系统环境中,增加一条记录,指向要加载的dll的所在目录。因为C++的代码中,Windows目录和Windows\System32目录以及环境变量设定的目录都是搜索路径之一。

这里提供怎么从C#中修改系统环境变量的代码:

static void AddEnvironmentPaths(IEnumerable<string> paths)
{var path = new[] { Environment.GetEnvironmentVariable("PATH") ?? string.Empty };
    string newPath = string.Join(Path.PathSeparator.ToString(), path.Concat(paths));
    Environment.SetEnvironmentVariable("PATH", newPath);
}

参考文章

  • 浅谈.NET中程序集的动态加载
  • 再谈CLR查找和加载程序集的方式,查找程序集
  • 深入理解CLR类加载机制
  • C#程序集加载方法

 

 

如果您觉得阅读本文对您有帮助,请点一下“推荐”按钮,您的“推荐”将是我最大的写作动力!如果您想持续关注我的文章,请扫描二维码,关注马非码的微信公众号,我会将我的文章推送给您,并和您一起分享我日常阅读过的优质文章。本文版权归作者和博客园共有,来源网址:http://www.cnblogs.com/marvin/欢迎各位转载,但是未经作者本人同意,转载文章之后必须在文章页面明显位置给出作者和原文连接,否则保留追究法律责

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436292.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【转】C#开发奇技淫巧二:根据dll文件加载C++或者Delphi插件

这两天忙着把框架改为支持加载C和Delphi的插件&#xff0c;来不及更新blog了。 原来的写的框架只支持c#插件&#xff0c;这个好做&#xff0c;直接用c#的反射功能便可。但是公司不是所有人都搞C#&#xff0c;也不是所有的程序C#都能很好的完成&#xff0c;又或者其他公司…

Linux编程练习 --多线程3--mutex

互斥指互相排斥的锁&#xff0c;是一种信号量&#xff0c;常用来防止两个进程或线程在同一时刻访问相同的共享资源 1.数据类型&#xff1a; 在Linux下, 线程的互斥量数据类型是pthread_mutex_t&#xff0c;我们定义一个互斥数据可以这样&#xff1a; pthread_mutex_t mutex; 2.…

java ppt转html_word,ppt,excel转pdf,pdf转html工具类搭建

我看到很多需求要求word,excel,ppt&#xff0c;pptx转pdf等工具类。还有就是pdf转图片转html这里介绍一个这个工具类。引入pom.xmlcom.asposeaspose-pdf11.0.0com.asposewords15.9.0com.asposeaspose-slides15.9.0工具类代码&#xff1a;package com.lvic.prsp.common.util;imp…

【转】一篇易懂的CAN通讯协议指南1

转自&#xff1a;https://zhuanlan.zhihu.com/p/162708070 本文力求以图文并茂来接好CAN通讯协议的基础知识&#xff0c;希望能给有兴趣的朋友带来一些收获。 为了便于大家理解与接受&#xff0c;请先看一幅图&#xff1a; 图1 电话会议 简单地讲CAN总线就如上图1中两根粗黄线…

长字串与短字串

procedure TForm1.Button1Click(Sender: TObject); vars1: ShortString; //为兼容老版本而存在s2: string; //长字串&#xff0c;相当于动态字符数组s3: string[10]; //短字串&#xff0c;相当于静态字符数组&#xff0c;代替 ShortString beginShowMessage(IntToStr(…

Linux编程练习 --多线程4--条件变量

上一篇练习了互斥锁的用法和原理&#xff0c;这次学习和互斥锁一起应用的cond--条件变量 1.互斥锁的存在问题&#xff1a; 互斥锁一个明显的缺点是它只有两种状态&#xff1a;锁定和非锁定。设想一种简单情景&#xff1a;多个线程访问同一个共享资源时&#xff0c;并不知道何时…

在PHP中实现中文汉字验证码

PHP代码<?php /********************************* * Code by Gently * 24/07/07 *严正声明&#xff1a;验证码为程序随机生成&#xff0c;“某种巧合”的词语组合属于正常现象&#xff0c; *某些别有用心的人不要借题发挥&#xff01; **************…

java哈夫曼编码译码_哈夫曼编码与译码 java

展开全部package qwp;import java.util.ArrayDeque;import java.util.ArrayList;import java.util.Collections;import java.util.HashSet;import java.util.List;import java.util.Queue;import java.util.Scanner;public class HuffmanCoding {public static String writeStr…

【转】预编译头文件

http://www.cnblogs.com/nzbbody/p/3437868.html 1、解决什么问题&#xff1f; C 编译器是单独、分别对每个cpp文件进行预编译&#xff08;也就是对#include&#xff0c;#define 等进行文本替换&#xff09;&#xff0c;生成编译单元。编译单元是一个自包含文件&#xff0c;C编…

Linux编程练习 --多线程5--信号量(semaphore)

这一篇练习信号量的应用 信号量本质上是一个非负的整数计数器&#xff0c;也是UNIX中古老的实现进程互斥和同步的手段&#xff0c;Linux下信号量概念是在线程中&#xff0c;信号则在进程控制中&#xff0c;不过原理差不多&#xff0c;最基本最经典的操作莫过于P、V操作了&#…

【转】CAN协议深度解析-简单易懂协议详解

转自&#xff1a;https://zhuanlan.zhihu.com/p/343607068 CAN接口兼容规范2.0A和2.0B(主动)&#xff0c;位速率高达1兆位/秒。它可以接收和发送11位标识符的标准帧&#xff0c;也可以接收和发送29位标识符的扩展帧。 扩展帧的仲裁域有29位&#xff0c;可以出现2^29中报文&…

深度神经网络下的风格迁移模型(C#)

版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请在显著位置标明本文出处以及作者网名&#xff0c;未经作者允许不得用于商业目的。 这个是C#版本的&#xff0c;这里就只放出代码。VB.Net版本请参看 深度神经网络下的风格迁移模型-CSDN博客 斯坦福大学李飞飞团队的…

使用 Visual Studio 2005 Team System 进行单元测试并生成用于 Unit Test Framework 的源代码...

PS&#xff1a;微软专家写的一往篇VSTS2005测试功能说明&#xff0c;非常详细。适合初学者查看。适用于&#xff1a;Microsoft Visual Studio 2005 Team System Beta 2Team Architect & Team Test EditionsMicrosoft Visual C# 2005摘要&#xff1a; Scott 详细介绍自动化单…

java银行利率_Java-银行计算利息

Bank类public class Bank {int savedMoney;int year;double interest;double interestRate0.29;public double computerInterest(){interestyear*interestRate*savedMoney;return interest;}public void setInterestRate(double rate){interestRaterate;}}ConstructionBank类pu…

【转】C++ win32窗口创建详解

转自&#xff1a;https://my.oschina.net/u/4328928/blog/3315324 本篇所讲解的内容仅限于 Windows 操作系统且限于 win32程序设计 现在我们在Windows系统上用的软件, 早已不是控制台界面, 而是窗体应用程序 窗体与控制台的区别就是: 有了窗口的概念 由于C的语法复杂, 使得…

java 子类继承父类_关于Java 的继承问题,子类会继承父类的哪些东西?-----转载...

和C类似&#xff0c;可以继承基类的公共属性和方法。在Java继承里&#xff0c;父类的属性还有方法在声明时&#xff0c;如果是public关键字即公共属性&#xff0c;则在子类继承时&#xff0c;这些属性和方法都会被子类继承。受保护的也可以继承但是私有的类属性成员和方法则无法…

天气很冷晚饭很香

今天和zhoujia 去吃她家门口的东北菜&#xff0c;席间谈笑风生&#xff0c;突然她说“亲爱的&#xff0c;眼光要看的长远一些”&#xff0c;附带标准的领导手势&#xff0c;让我笑翻。zhoujia是个乐观的人&#xff0c;从来没有见她发愁过&#xff0c;她的房子装修的也差不多了&…

【转】win32窗口的大小,居中,拖动

参考 https://www.cnblogs.com/findumars/p/3948315.html 不让调整大小 窗口风格设置&#xff0c;后面两个是最大最小化按钮 dwStyle^WS_THICKFRAME^WS_MAXIMIZEBOX^WS_MINIMIZEBOX 不让拖动 消息循环里添加 case WM_NCLBUTTONDOWN: { switch (wParam) …

java +号变空格_base64码通过http传输 +号变 空格 问题解决

通过七牛云base64上传图片&#xff0c;通过官方示例上传成功后&#xff0c;根据示例改了一个controller。通过前端往后端传base64码形式进行测试。死活不通过&#xff0c;七牛报400。仔细排查后发现&#xff0c;示例转换的base64码与前端传来的base64码稍有区别&#xff0c;前端…

Linux编程练习 --进程间通信1--无名管道

进程间通信系列--管道 管道可以说是最古老的IPC形式&#xff0c;所谓管道&#xff0c;是指进程间建立的一条通信的通道&#xff0c;从本质上看&#xff0c;管道是UNIX文件概念的推广管道通信的介质是文件&#xff0c;先看一下管道的特点&#xff1a; 1.管道的特点&#xff1a; …