引用原文網(wǎng)址:http://www.moon-soft.com/doc/33045.htm
程序集 與 托管模塊
如果你正在開(kāi)發(fā)面向DotNet平臺(tái)的應(yīng)用程序,那么你肯定對(duì)“程序集”和“托管模塊”這兩個(gè)概念不陌生,這是DotNet帶來(lái)的術(shù)語(yǔ)。這兩個(gè)概念很容易混淆,有人認(rèn)為它們指的是同一樣事物,其實(shí)不然。這里,我寫下自己的一些理解。
為了便于說(shuō)明問(wèn)題,我們先看看一個(gè)公司的某項(xiàng)目開(kāi)發(fā)團(tuán)隊(duì),這個(gè)“團(tuán)隊(duì)”由團(tuán)隊(duì)“成員”組成(可能還有一些資源),而在這些團(tuán)隊(duì)成員中,必然有一個(gè)成員是這個(gè)團(tuán)隊(duì)的頭頭,為了表示不是普通的成員,頭頭都有一個(gè)稱號(hào),比如說(shuō)“經(jīng)理”或者“負(fù)責(zé)人”。經(jīng)常的,說(shuō)到某個(gè)團(tuán)隊(duì),我們只要知道它的頭頭就可以了。也就是說(shuō),頭頭代表了這個(gè)團(tuán)隊(duì)(注意,這里是"代表")。如果其他人有什么事情要與這個(gè)團(tuán)隊(duì)交流(比如團(tuán)隊(duì)向外說(shuō)明本團(tuán)隊(duì)完成了什么任務(wù)等),沒(méi)有必要將每個(gè)成員找來(lái)交流,我們只要找到頭頭就可以了。 因?yàn)轭^頭知道本團(tuán)隊(duì)的一切,他掌握了關(guān)于團(tuán)隊(duì)的所有信息:本團(tuán)隊(duì)有哪些成員,而且有些什么資源可以利用,與其他哪些團(tuán)隊(duì)有交流等等。常見(jiàn)的團(tuán)隊(duì)有多個(gè)成員,一個(gè)成員的團(tuán)隊(duì)很少見(jiàn)(頭頭也是他),但為了說(shuō)明問(wèn)題,我們假定一個(gè)成員的團(tuán)隊(duì)和多個(gè)成員的團(tuán)隊(duì)都存在。
從上面的討論中我們可以這樣總結(jié):團(tuán)隊(duì)是個(gè)邏輯概念,并不是指某個(gè)人或資源,它是一個(gè)集合,而且這個(gè)集合不為空,只有當(dāng)擁有大于或者等于多個(gè)成員的時(shí)候它才稱之為團(tuán)隊(duì)。
嘿,大家看出來(lái)了,我之所以要說(shuō)項(xiàng)目開(kāi)發(fā)團(tuán)隊(duì)和成員,是因?yàn)樵?/span>DotNet中的程序集和托管模塊的概念與此類似。程序集其實(shí)并不是說(shuō)某個(gè)文件,它是一個(gè)邏輯概念,就像一個(gè)團(tuán)隊(duì)。當(dāng)然,我們習(xí)慣說(shuō)a.exe 或者b.dll是個(gè)程序集,其實(shí)這樣說(shuō)多少會(huì)讓人混淆,請(qǐng)看下面的說(shuō)明。而托管模塊就像是團(tuán)隊(duì)中的團(tuán)隊(duì)成員。哈,你肯定想到了,其中必然有一個(gè)托管模塊是整個(gè)“團(tuán)隊(duì)”的“頭頭”,它負(fù)責(zé)管理所有的托管模塊,關(guān)于這個(gè)程序集的一些信息也保存在這個(gè)托管模塊中。我們把這樣的模塊叫做“主托管模塊”(我記得某本書上是這樣叫的)。為了區(qū)分主托管模塊和普通托管模塊,怎么辦呢?這個(gè)好說(shuō),人最擅長(zhǎng)給自己掛頭銜了,借鑒一下。我們給普通托管模塊和主托管模塊不同的后綴名,普通托管模塊的后綴名是.netmodule,而主托管模塊的后綴名是.exe 或者.dll。和開(kāi)發(fā)團(tuán)隊(duì)中頭頭代表整個(gè)團(tuán)隊(duì)一樣,主托管模塊代表著整個(gè)程序集。既然是主托管模塊,那么它肯定與其他托管模塊不一樣。一般的托管模塊包含有IL代碼和元數(shù)據(jù),而這個(gè)主托管模塊有沒(méi)有IL代碼和元數(shù)據(jù)并不重要,但它一定要有“清單”,也就是關(guān)于整個(gè)程序集的數(shù)據(jù)。
為什么主托管模塊會(huì)有兩種后綴名呢?這是因?yàn)榉止さ脑颍械某绦蚣菫槠渌绦蚣峁┍憷模鼘?shí)現(xiàn)了一些數(shù)據(jù)類型(類庫(kù))或者擁有某些資源――這就是后綴名為.dll的程序集。這樣的程序集不喜歡拋頭露面,它們是“幕后程序集”。另一種程序集,后綴名為.exe,用戶直接和它們打交道。有的功能它自己實(shí)現(xiàn),但有些呢就交給幕后程序集去實(shí)現(xiàn)。為了區(qū)別,我們就把這樣的程序集叫做“前臺(tái)程序集”吧。DotNet規(guī)范中沒(méi)有這兩個(gè)概念,這里是為了理解才造的詞。當(dāng)然,讀者朋友都會(huì)有自己比較習(xí)慣的理解名字。一個(gè)可執(zhí)行應(yīng)用程序中,只有一個(gè)前臺(tái)程序集的概念,但是可以有零個(gè)或一個(gè)或多個(gè)幕后程序集。
我們又回到實(shí)際生活中的開(kāi)發(fā)團(tuán)隊(duì)概念上來(lái)。團(tuán)隊(duì)的組成有這樣幾種情況:
a.一個(gè)成員,這樣的成員是比較厲害的那種,比如個(gè)人軟件開(kāi)發(fā)者,自己對(duì)自己負(fù)責(zé),自己是自己的頭頭;
b.多個(gè)成員,其中某個(gè)成員是頭頭;
程序集與此類似,所以就有了單模塊和多模塊程序集(有時(shí)也稱單文件程序集和多文件
程序集)。如果要實(shí)現(xiàn)的功能不是很煩雜,那么就用單模塊程序集吧;但是,如果要實(shí)現(xiàn)的功能比較多,而且也好分開(kāi),那么建議你用多模塊程序集吧,如果以后某個(gè)功能的實(shí)現(xiàn)方案改變的話,只要修改這個(gè)模塊就行了,這樣一來(lái),極大的降低了開(kāi)發(fā)的煩雜度。還有幾點(diǎn)要注意,有的托管模塊并不實(shí)現(xiàn)某種計(jì)算功能,它們僅僅是提供一些資源,比如說(shuō)字符串;有時(shí)程序集除了包含托管模塊外,還包含一些另外的資源文件。上面說(shuō)的好像不是很直觀,我們?cè)俳Y(jié)合下面的幾幅圖來(lái)理解。
--------------------------------------
--------------------------------------
圖1。某開(kāi)發(fā)團(tuán)隊(duì)
--------------------------------------
------------------------------------
圖2。某程序集示意圖
我相信現(xiàn)在大家對(duì)程序集和托管模塊分別是什么以及兩者間的關(guān)系有了較好的理解。但是如果有源代碼輔助一下那就更好了。是的。對(duì)程序員來(lái)說(shuō),源代碼比什么都親切。好的,下面就舉兩個(gè)簡(jiǎn)單的例子(用C#語(yǔ)言表述)。一個(gè)是單模塊程序集,一個(gè)是多模塊程序集。兩者都是前臺(tái)程序集(后綴名是exe)。我用Visual C# 2003 集成開(kāi)發(fā)環(huán)境試了一下,竟然發(fā)現(xiàn)它不支持多程序集的開(kāi)發(fā)(希望是我沒(méi)有找到)。沒(méi)關(guān)系,我們還有DotNet FrameWork SDK呢,不用怕。它自帶的C#編譯器csc.exe很好用。至于csc的用法我就不多說(shuō)了。
例子1。單模塊程序集:
首先找個(gè)文本編輯器,將下面的代碼敲進(jìn)去,然后將它保存起來(lái),取名為hello.cs。 例如我把它存為e:\test\hello.cs。
/*
*hello.cs 在控制臺(tái)上顯示一行字符串
*/
using System;
namespace nsApp
{
public class CEnterPoint
{
static void Main()
{
Console.WriteLine("Hello, verybody!");
}
}
}
然后,打開(kāi)SDK命令行提示,定位在hello.cs所在的文件夾,比如我的是e:\test。在命令行提示中輸入命令 csc hello.cs,這個(gè)命令默認(rèn)的是生成 .exe文件,也就是一個(gè)主托管模塊。 現(xiàn)在到hello.cs所在的文件夾中看看,你會(huì)發(fā)現(xiàn)一個(gè)新文件hello.exe。好,一個(gè)主托管模塊誕生了。既然有主托管模塊,那么就標(biāo)志著一個(gè)程序集的誕生。再看看,沒(méi)有其他的托管模塊,那么這個(gè)程序集就是一個(gè)單模塊程序集。
例子2。多模塊程序集:
假設(shè)我們要編一個(gè)程序來(lái)管理動(dòng)物園。這里為了簡(jiǎn)單,不考慮繼承等問(wèn)題,每個(gè)動(dòng)物用一個(gè)類來(lái)表示,并且用一個(gè)托管模塊來(lái)實(shí)現(xiàn)。將所有的動(dòng)物都放在命名空間nsZoo中。現(xiàn)在動(dòng)物園有兩種動(dòng)物:Dog和Cat。看下面的代碼:
/* Dog.cs */
using System;
namespace nsZoo
{
public class Dog
{
public void SayHello()
{
Console.WriteLine("I am a dog, wang wang wang");
}
}
}
/* Cat.cs */
using System;
namespace nsZoo
{
public class Cat
{
public void SayHello()
{
Console.WriteLine("I am a cat, miao miao miao");
}
}
}
/* Main.cs */
using System;
using nsZoo;
namespace nsApp
{
class CEnterPoint
{
static void Main(string[] args)
{
Dog aDog = new Dog();
Cat aCat = new Cat();
aDog.SayHello();
aCat.SayHello();
Console.ReadLine();
}
}
}
將上面的三個(gè)文件保存好,比如放在文件夾e:\zoo 中。再一次打開(kāi)SDK命令行提示,定位到e:\zoo
輸入命令 csc /t:module Dog.cs 看看文件夾,你會(huì)發(fā)現(xiàn)一個(gè)新文件 Dog.netmodule
哈,這是一個(gè)托管模塊,普通的托管模塊。
接著輸入命令 csc /t:module Cat.cs 同樣得到一個(gè)普通托管模塊 Cat.netmodule 。好現(xiàn)在有了兩個(gè)普通托管模塊。
為了實(shí)現(xiàn)一個(gè)程序集的夢(mèng)想,必須還要有一個(gè)主托管模塊。好,接著輸入
csc /addmodule:Dog.netmodule;Cat.netmodule Main.cs 你會(huì)在文件夾中發(fā)現(xiàn)一個(gè)名為Main.exe的文件,這就是你想要的主托管模塊。
好了。現(xiàn)在你有三個(gè)托管模塊,并且任命其中一個(gè)為主托管模塊,這樣你就擁有了一個(gè)程序集了。這個(gè)程序集的組成是:Dog.netmodule, Cat.netmodule, Main.exe 。我們習(xí)慣說(shuō)這個(gè)程序集是Main.exe 。但是一定要知道,Main.exe 其實(shí)是一個(gè)程序集的“頭頭”,由它代表著整個(gè)程序集。