原来上网我下了很多VS2003制作安装Demo。而我用的是VS2005,我在使用这些小样时,基本上绝大部分基于VS2003制作安装工程都因为少数文件无法转换而无法加载工程,为此我也苦恼很久,后来终于让我找到了解决办法。
1.新建一个空解决方案这个解决方案是为了容纳下面几个项目。
2.新建类库项目在解决方案下新建类库项目,这个类库项目就是要生成的程序集,它会被编译为一个独立的dll。但是要把一个程序集放入GAC中,需要给这个程序集赋予一个强名称的签名
3.生成加密密钥对
Sdk带有一个生成强名称程序集签名用的密钥对的命令sn.exe(可以在C:\Program Files\Microsoft Visual Studio .NET 2003\SDK\v1.1\Bin目录下找到),使用下面的命令生成一个密钥对叫做Testkey.snk,命令如下:
sn -k Testkey.snk
#t#将生成密钥对拷贝到类库项目的根目录下。
编辑类型项目的AssemblyInfo.cs文件,将密钥文件属性指向上面那个密钥对文件
[assembly: AssemblyKeyFile("..\\..\\Testkey.snk")]
这里的路径是相对于编译后的项目输出的位置。
AssemblyInfo.cs中其他的设置根据实际需要设置即可。
一个带有强名称的程序集设计即告完成
4.新建主项目
一般情况都是一个exe的主项目引用一个或者多个dll的类库项目,我们在解决VS2003制作安装方案中新建一个主项目,引用上面步骤建立的类库项目。
项目实现你自己想做的功能。
5.新建一个部署项目
在解决方案中新建一个部署项目。
在部署项目的“文件系统”中,点击右键“添加特殊文件夹”中选择“Global Assembly Cache Folder文件夹”,这样在文件系统中新建了一个Global Assembly Cache Folder文件夹,这个文件夹指向系统的GAC目录。
添加项目输出,选择主项目的项目主输出,主项目的输出会自动加入到“应用程序文件夹”。
再添加项目输出,选择类库项目的主输出,同样类库项目的dll也会被加入到“应用程序文件夹”。我们把这个dll从“应用程序文件夹”拖到“Global Assembly Cache Folder文件夹”。这样系统安装时会把这个dll安装到GAC了。
6.最后设置程序目录和桌面上是否要放置程序的输出项
7.build部署项目,生成msi
8.在目标机器上安装这个msi,安装完毕会在C:\WINDOWS\assembly找到刚才类库中生成的dll,已经作为一个全局程序集加入到GAC中。在GAC中同时能够看到这个dll的版本,语言和强名称中的public key的一部分public key token。
在整个过程中有几点需要特别注意:
1.一般的软件我们都希望能够在出了新版本后生成msi在客户机上安装能够自动删除原先的版本后安装新版本。同样vs2003也提供了这样的选项,在部署项目属性中RemovePreviousVersion属性设置为true,指示系统在安装之前删除老版本。
2.如果一个部署项目上ProductName和应用文件夹的DefaultLocation不变,则会被认为是同一个程序。部署项目的version和ProductCode和UpgradeCode改变后会被认为同一个程序的版本升级。如果只改变了version没有同时更改ProductCode和UpgradeCode,会被系统认为还是同一个版本的程序,在已有了原来系统的机器上不能再次安装。
3.项目中安装到GAC的dll项目也会同整个软件一起被安装被卸载,但是这个dll的版本跟部署项目设置的版本不同,删除时系统会根据msi中带的dll的程序集的完整标识即程序集名称、version、cultrue、key token四项来检查。当msi中的dll的四项跟在GAC中的dll有一项不同,则被认为是不同版本的dll,在升级安装时不会删除原来的dll,同时将新版本的dll加入到GAC。
4.在部署项目中加入到GAC的dll需要有个特殊的设置,就是要把这个dll的属性中Permanent设置为ture,表示卸载应用程序时是否移除项目输出文件。这里一定要设为true,在每次更新安装时会将没有改动的dll先从gac删除然后再安装进去。如果没设为true,程序更新安装时不会先将gac中的这个dll卸掉,安装时会把这个dll再安装一次,结果发现gac中有一个跟自己一模一样的dll,msi就会执行一个删除操作,把这个dll从gac中删除。最后的结果就是,项目中的其他部分都更新了,但是这个dll却没有了,导致系统不能正常使用。
奇怪的是,项目中其它的文件不需要将Permanent设置为ture,它也会在升级安装时先删除文件,然后安装新的文件。