C#反射內(nèi)存的處理分析
本文實(shí)例分析了C#反射內(nèi)存的處理。分享給大家供大家參考。具體分析如下:
這段時(shí)間由于公司的項(xiàng)目的要求,我利用c#的反射的機(jī)制做了一個(gè)客戶端框架??蛻舳死锏乃械哪K都是以一定形式進(jìn)行提供,例如:FORM,UserControl. 在做的過程中很簡單與愉快。具體的過程如下:
1. 收集客戶的需求
2. 整理需求,形成必要的文檔
3. 通過討論大體的得到程序的界面風(fēng)格
4. 由UI設(shè)計(jì)師設(shè)計(jì)出來具體的界面形式
5. 通過需求封裝必要的服務(wù)(我們可以使用c#的WCF服務(wù)或者JAVA的服務(wù))
6. 制作服務(wù)管理框架
7. 封裝程序要使用到的控件
8. 編寫客戶端框架
9. 編寫模塊
10. 加載進(jìn)行測試
上面說的就是簡單的一個(gè)開發(fā)的過程,當(dāng)然里面包括了很多的汗水。一個(gè)好的程序都要滿足最基本的可卸載,可插入。即插件式架構(gòu)。無論是客戶端,還是服務(wù)端都要采用插件式管理。
在做c#客戶端框架的時(shí)候,利用微軟的反射與工廠模式的機(jī)制的時(shí)候,里面有個(gè)很大的問題。就是通過反射的DLL加載到內(nèi)存中的時(shí)候無法進(jìn)行內(nèi)存的釋放,只有你關(guān)閉程序的時(shí)候才進(jìn)行內(nèi)存的釋放,這點(diǎn)有很大的缺陷。我在網(wǎng)上也找了很多的解決的辦法,但是沒有一個(gè)能夠成功的。其中最經(jīng)典的是插件的卸載的方式,這種方式我也進(jìn)行的實(shí)驗(yàn),雖然能夠釋放部分內(nèi)存,但是不能釋放全部的內(nèi)存。我和很多程序員聊這個(gè)事情的時(shí)候,他們說把一切能釋放的都釋放掉。但是你就算做到這些也不能做到很好的釋放效果(也許的我的水平不行)。今天來吐槽一下VS的內(nèi)存的釋放。VS的內(nèi)存都是通過托管的機(jī)制進(jìn)行資源的使用與釋放,對(duì)于非托管資源可以通過析構(gòu)函數(shù)與其他的方式進(jìn)行釋放。對(duì)于反射的情況微軟沒有給一個(gè)很好的辦法。如果程序員兄弟們有好的辦法提供給我們學(xué)習(xí)那將是個(gè)大的善果。
我在上面說過通過卸載插件的方式是可以釋放部分的內(nèi)存,效果也還行,但是對(duì)于一些WCF服務(wù)寫的控件,在通過遠(yuǎn)程的模式確實(shí)存在一些問題。具體的部分實(shí)現(xiàn)代碼如下:
{
#region class-level declarations
private Assembly a = null;
#endregion
#region constructors and destructors
public AssemblyLoader(string fullPath)
{
if (a == null)
{
a = Assembly.LoadFrom(fullPath);
}
}
~AssemblyLoader()
{
dispose(false);
}
public void Dispose()
{
dispose(true);
}
private void dispose(bool disposing)
{
if (disposing)
{
a = null;
System.GC.Collect();
System.GC.WaitForPendingFinalizers();
System.GC.Collect(0);
}
}
#endregion
#region public functionality
public object GetObject(string typename, object[] ctorParms)
{
BindingFlags flags = BindingFlags.CreateInstance | BindingFlags.Instance | BindingFlags.Public;
object o = null
;
if (a != null)
{
try
{
o = a.CreateInstance(typename, true, flags, null, ctorParms, null, null);
}
catch
{
}
}
return o;
}
public object GetObject(string typename)
{
return GetObject(typename, null);
}
#endregion
public class ObjectLoader : IDisposable
{
// essentially creates a parallel-hash pair setup
// one appDomain per loader
protected Hashtable domains = new Hashtable();
// one loader per assembly DLL
protected Hashtable loaders = new Hashtable();
public ObjectLoader()
{}
public object GetObject(string dllName, string typeName, object[] constructorParms)
{
AssemblyLoader al = null;
object o = null;
//Type t = null;
try
{
al = (AssemblyLoader)loaders[dllName];
}
catch (Exception) { }
if (al == null)
{
AppDomainSetup setup = new AppDomainSetup();
setup.ShadowCopyFiles = "true";
AppDomain domain = AppDomain.CreateDomain(dllName, null, setup);
int key=0;
foreach (DictionaryEntry de in domains)
{
if(de.Key.ToString()==dllName)
{
key++;
break;
}
}
if (key == 0)
{
domains.Add(dllName, domain);
}
object[] parms = { dllName };
BindingFlags bindings = BindingFlags.CreateInstance | BindingFlags.Instance | BindingFlags.Public;
try
{
//al = (BCFrameWork.Client.ClientInfrm.AssemblyLoader)domain.CreateInstanceFromAndUnwrap(
// "Loader.dll", "Loader.AssemblyLoader", true, bindings, null, parms, null, null, null);
al = (AssemblyLoader)domain.CreateInstanceFromAndUnwrap(
"BestelLoadDll.dll", "BestelLoadDll.AssemblyLoader", true, bindings, null, parms, null, null, null);
}
catch
{
}
if (al != null)
{
if (!loaders.ContainsKey(dllName))
{
loaders.Add(dllName, al);
}
}
}
if (al != null)
{
o = al.GetObject(typeName, constructorParms);
}
return o;
}
public void Unload(string dllName)
{
if (domains.ContainsKey(dllName))
{
AppDomain domain = (AppDomain)domains[dllName];
AppDomain.Unload(domain);
domains.Remove(dllName);
}
}
~ObjectLoader()
{
dispose(false);
}
public void Dispose()
{
dispose(true);
}
private void dispose(bool disposing)
{
if (disposing)
{
loaders.Clear();
List removeobj = new List();
foreach (object o in domains.Keys)
{
string dllName = o.ToString();
removeobj.Add(dllName);
}
foreach (string item in removeobj)
{
Unload(item);
}
domains.Clear();
System.GC.Collect();
}
}
}
調(diào)用方式很簡單,如果你了解反射就知道怎么調(diào)用,這個(gè)寫法能夠滿足普通的用戶控件的反射遠(yuǎn)程加載,但是對(duì)于一些特殊的用戶控件還是沒有辦法。
希望本文所述對(duì)大家的C#程序設(shè)計(jì)有所幫助。
相關(guān)文章
C# Lambda表達(dá)式select()和where()的區(qū)別及用法
這篇文章主要介紹了C# Lambda表達(dá)式select()和where()的區(qū)別及用法,select在linq中一般會(huì)用來提取最后篩選的元素集合,在lambda表達(dá)式中通常用where得到元素集合,需要的朋友可以參考下2023-07-07C#中的小數(shù)和百分?jǐn)?shù)計(jì)算與byte數(shù)組操作
這篇文章介紹了C#中的小數(shù)和百分?jǐn)?shù)計(jì)算與byte數(shù)組操作,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04C# 指針內(nèi)存控制Marshal內(nèi)存數(shù)據(jù)存儲(chǔ)原理分析
這篇文章主要介紹了C# 指針 內(nèi)存控制 Marshal 內(nèi)存數(shù)據(jù)存儲(chǔ)原理分析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02使用MSScriptControl 在 C# 中讀取json數(shù)據(jù)的方法
下面小編就為大家?guī)硪黄褂肕SScriptControl 在 C# 中讀取json數(shù)據(jù)的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-01-01