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