AppDomain加载与释放dll

几年前写过同名随笔,但今天应不大适用了。但还有几个朋友留言关注,我重新发布相关代码。

首先我们的目的就是运行期间更新dll,并应用dll。这个过程需要应用 AppDomain来解决,办法就是创建新的AppDomain加载dll,并在dll更新后创建新的AppDomain来加载新的dll。

好了,我们有两个目录需要完成。

第一个目标:运行时能更新dll。

1.创建AppDomain时设置ShadowCopyFiles

AppDomainSetup _setup =new AppDomainSetup(); _setup.ShadowCopyFiles ="true"; _setup.LoaderOptimization = LoaderOptimization.SingleDomain; AppDomain _curDomain = AppDomain.CreateDomain("TestDomain", null, _setup);

2.当我们的主程序需要引用动态加载dll中的某个对象的实例时,该实例必须创建为代理类。代理类的意思是类需要继承自MarshalByRefObject。此外一般会使用另一个公共类库定义基类或接口来使用这个代理类。好了,先看怎么样在前面创建的程序域中创建代理类的实例

System.Runtime.Remoting.ObjectHandle objHandle = pJobDomain.CreateInstanceFrom(strDllFilePath, strClassName); object objBuild = objHandle.Unwrap();
if (objBuild ==null) thrownew Exception("buildWorker -- 创建对象失败"); else ido = (IDo)objBuild;//IDo是一个接口

好了,如果strDllFilePath=“ClassLibrary.dll”,那么运行目录的ClassLibrary.dll文件被加载后,我们还是可以随便更新他的。但更新后,程序域不重新创建的话,使用上面的方法再次创建代理实例,我们的程序依然看不到新的dll的运行效果的,因为我们加载时设置了ShadowCopyFiles="true"。当然你可能会说设置为"false"行不行呢?你可以试试,但我要说的是,如果这样的话,我们第一个目标都完成不了。

第二个目标:应用新的dll。

其实很简单,UnLoad后再创建一次,使用新的AppDomain创建代理类 ,程序就能看到新dll的效果。好,我定义了一个方法保证每运行一次都会创建一个新的程序域

AppDomain _curDomain; privatevoid load() {     if (_curDomain !=null)//已创建了程序域    {         AppDomain.Unload(_curDomain);//先关闭再打开。        _curDomain =null;     }                 AppDomainSetup _setup =new AppDomainSetup();     _setup.ShadowCopyFiles ="true";                 _setup.LoaderOptimization = LoaderOptimization.SingleDomain;         _curDomain = AppDomain.CreateDomain("TestDomain", null, _setup); }

提供一个kevin-y的测试的源代码 给大家下载

posted @ 2011-03-12 11:49 生命体验之kevin-Y 阅读(455) 评论(3)  编辑

“创建窗口句柄时出错” -- 不要过份相信垃圾回收

我总相信只要对象离开其作用域并没有被引用到,dotnet会帮我们回收他,当然一般还会告诉我们IO操作(网络也可视为IO操作的一类)需要主动关闭。而我一直忽略了控件也是需要这么做的。

工作让我接触到了一个很古怪的问题,客户反馈说每天有一台机器到了11点就会出错。观察并细听用者的反应,我认为与使用次数有关。界面动态不断地加载不定数量的用户控件。起初怀疑是不是反射在某些应用下会引起问题,于是写了个项目模拟实际的应用,测试以下三种办法创建一个用户控件的差异

///<summary>/// 方式一:全程反射  ///</summary>///<returns></returns>private UserControl createUserControl()  {      Assembly asm = Assembly.Load("ClassLibrary");      UserControl ctl = asm.CreateInstance("ClassLibrary.MyEditContro") as UserControl;      return ctl;  }  // 缓存Assembly Assembly asmOne = Assembly.Load("ClassLibrary");  ///<summary>/// 方式二:缓存Assembly  ///</summary>///<returns></returns>private UserControl createUserControlOne()  {      UserControl ctl = asmOne.CreateInstance("ClassLibrary.MyEditContro") as UserControl;      return ctl;  }  ///<summary>/// 方式三:不使用反射  ///</summary>///<returns></returns>private UserControl createMySelf()  {      ClassLibrary.MyEditContro ctl =new ClassLibrary.MyEditContro();      return ctl;  }      

好了,我要不断的创建ClassLibrary.MyEditContro 控件并添加到一个Pannel中显示,以下是测试的主体方法

privatevoid runTest(int pMax,int pMethod) {     curNo =;     for (int i =; i < pMax; i++)     {         curNo++;         UserControl ctl =null;         switch (pMethod)         {             case:                 ctl = createUserControl();                 break;             case:                 ctl = createUserControlOne();                 break;             case:                 ctl = createMySelf();                 break;         }                             ctl.Dock = DockStyle.Fill;         this.panel1.Controls.Clear();         this.panel1.Controls.Add(ctl);         this.richTextBox1.AppendText(string.Format("{0}.完成测试.\r\n", curNo));         Application.DoEvents();             } }

结果三种方法,在创建400多次后都报以下异常

System.ComponentModel.Win32Exception: 创建窗口句柄时出错。    在 System.Windows.Forms.NativeWindow.CreateHandle(CreateParams cp)    在 System.Windows.Forms.Control.CreateHandle().....

如何解决了?去除不断创建控件的方式,只保留一个实例?其实我认为这是最理想的。不过解决问题应优先从最简便有效的方式去考虑,同事想到了在UserControl使用完后Dispose,于是在Application.DoEvents后加上一句“ctl.Dispose();”

... this.panel1.Controls.Clear(); this.panel1.Controls.Add(ctl); this.richTextBox1.AppendText(string.Format("{0}.完成测试.\r\n", curNo)); Application.DoEvents(); ctl.Dispose();//控件使用完后Dispose。主体测试方法增加这一句。

最后结果是三个方法都顺利通过1000次的创建显示。

AppDomain加载与释放dll的更多相关文章

  1. C#.Net 如何动态加载与卸载程序集(.dll或者.exe)0-------通过应用程序域AppDomain加载和卸载程序集

    本博客中以“C#.Net 如何动态加载与卸载程序集(.dll或者.exe)”开头的都是引用莫问奴归处 微软装配车的大门似乎只为货物装载敞开大门,却将卸载工人拒之门外.车门的钥匙只有一把,若要获得还需要 ...

  2. C#.Net 如何动态加载与卸载程序集(.dll或者.exe)2----通过应用程序域AppDomain加载和卸载程序集之后,如何再返回原来的主程序域

    实现目的:动态加载dll,执行完毕之后可以随时卸载掉,并可以替换这些dll,以在运行中更新dll中的类. 其实就是通过应用程序域AppDomain加载和卸载程序集. 在这方面微软有篇文章http:// ...

  3. C#中动态加载和卸载DLL

    在C++中加载和卸载DLL是一件很容易的事,LoadLibrary和FreeLibrary让你能够轻易的在程序中加载DLL,然后在任何地方卸载.在C#中我们也能使用Assembly.LoadFile实 ...

  4. 其原因可能是堆被损坏,这说明**.exe中或它加载的任何DLL中有Bug

    最近在写一个写日志文件的线程时,调用了HeapAlloc/HeapFree 申请/释放堆缓冲内存.调用HeapFree释放有个条件就是,日志的空闲缓冲队列中内存块超过100个.在测试的时候,发现调用H ...

  5. C#.Net 如何动态加载与卸载程序集(.dll或者.exe)1----C#中动态加载和卸载DLL

    我们知道在C++中加载和卸载DLL是一件很容易的事,LoadLibrary和FreeLibrary让你能够轻易的在程序中加载DLL,然后在任何地方卸载. 在C#中我们也能使用Assembly.Load ...

  6. 动态加载EXE和DLL

    程序中加载了一个DLL文件,但生成的EXE在脱离了DLL文件后仍然可以 单独使用,这是动态加载DLL技术.即:调用资源中的DLL. 此技术的好处:EXE可以使用DLL中的函数,但不会额外增加一 个DL ...

  7. 通过应用程序域AppDomain加载和卸载程序集

    微软装配车的大门似乎只为货物装载敞开大门,却将卸载工人拒之门外.车门的钥匙只有一把,若要获得还需要你费一些心思.我在学习Remoting的时候,就遇到一个扰人的问题,就是Remoting为远程对象仅提 ...

  8. c# 动态加载和卸载DLL程序集

    原文:c# 动态加载和卸载DLL程序集 在 C++中加载和卸载DLL是一件很容易的事,LoadLibrary和FreeLibrary让你能够轻易的在程序中加载DLL,然后在任何地方卸载.在 C#中我们 ...

  9. module_init宏解析 linux驱动的入口函数module_init的加载和释放

    linux驱动的入口函数module_init的加载和释放 http://blog.csdn.net/zhandoushi1982/article/details/4927579 void free_ ...

随机推荐

  1. POJ 2112: Optimal Milking【二分,网络流】

    题目大意:K台挤奶机,C个奶牛,每台挤奶器可以供M头牛使用,给出奶牛和和机器间的距离矩阵,求所有奶牛走最大距离的最小值 思路:最大距离的最小值,明显提示二分,将最小距离二分之后问题转化成为:K台挤奶机 ...

  2. leetcode之twosum

    class Solution { public: vector<int> twoSum(vector<int>& nums, int target) { vector& ...

  3. BZOJ1775: [Usaco2009 Dec]Vidgame 电视游戏问题

    n<=50个游戏机有花费,每个游戏机有Gi<=10种游戏,每种游戏有花费有收益,买了游戏机才能玩对应游戏,求最大收益. 这就是一个背包!不过有依存关系,就不会了! 方法一:f[i][j]表 ...

  4. 9.6——string类型

    string: getline(is,s):从输入流is读入到字符串s中 s1+s2:将两个字符串连接起来 构造string一些方法: 1)string s(cp,n):将s初始化为cp所指的n个字符 ...

  5. iOS 如何查看崩溃日志

    参考网址: [转载]https://www.jianshu.com/p/4de55d73c82b [转载]https://blog.csdn.net/qq_26544491/article/detai ...

  6. jree-创建普通折线图

    对于maven工程,需要引入依赖:在pom.xml中,添加如下内容 <dependency> <groupId>jfree</groupId> <artifa ...

  7. C++ std::tr1::bind使用

    1. 简述 同function函数相似.bind函数相同也能够实现相似于函数指针的功能.但却却比函数指针更加灵活.特别是函数指向类 的非静态成员函数时.std::tr1::function 能够对静态 ...

  8. 【python】super()

    转自: http://www.cnblogs.com/lovemo1314/archive/2011/05/03/2035005.html

  9. [RxJS] Implement RxJS `concatMap` by Waiting for Inner Subscriptions to Complete

    Unlike mergeMap and switchMap, concatMap focuses on when "inner" subscriptions "compl ...

  10. SQL 约束(Constraints)

    SQL 约束(Constraints) SQL 约束(Constraints) SQL 约束用于规定表中的数据规则. 如果存在违反约束的数据行为,行为会被约束终止. 约束可以在创建表时规定(通过 CR ...