[转].NET程序在windows操作系统上独立运行的技术要点

时间:2023-03-08 16:09:44
发现一个不错的网站,转载一篇文章方便查看
===============================================
    作者:宇内流云

最让.NET程序员苦恼的是,辛辛苦苦写出来的.NET程序,需要客户机上安装了.NET才能运行。仅为一个小小的应用程序去下载上百兆的.NET安装包,还得把它老老实实安装到客户机上,并占掉数百兆磁盘空间,这无疑是一件得不偿失的事情。.NET程序的这个弱点,也是影响.NET应用程序普及和价值的一个重要因素。
    所谓“独立运行”,是指.NET应用程序脱离完整的.NET运行环境,像c语言编译的程序那样,在操作系统上直接运行。简单地说就是:客户电脑无需安装任何版本的.NET框架,你的.NET程序照样可以在他的电脑或服务器上运行。
    .NET程序独立运行的基础是mono运行时以及它的程序集。mono是什么呢,mono是一款开源、免费、可定制的跨平台.NET运行环境,同时,它还包含了一系列具有重要意义的实用工具,当前最新的版本号是3.0.10,本文所采用的mono,即是这个版本号的windows版。

那么,到底怎么才能让你的.NET程序无障碍地在没有安装.NET平台的客户机“独立运行”呢,下面直奔主题。

一,建立跨平台的.NET环境与编译环境:
    1、下载并安装mono的windows版,建议将它安装到c:\mono文件夹中。
    2、安装cygwin。
       A、建议将它安装到c:\cygwin文件夹中。
       B、安装时,请将mingw-gcc、mingw-zlib、pkg-config、libiconv这几个组件选上,这是将.NET程序转化为本地程序的必要的编译环境。

二,启动cgywin并设置环境变量:
    1、点击开始菜单或桌面上的cygwin图标,启动且进入cygwin环境。
    2、输入下面的命令,设置或修改必要的环境变量:
       export PKG_CONFIG_PATH=/cygdrive/c/mono/lib/pkgconfig
       export PATH=$PATH:/cygdrive/c/mono/bin

三,将你的.NET程序转化为“独立程序”

请注意,这是本文的关键所在,很多地方的操作都有别于其它网文和mono官网所介绍的操作技术。

1,复制文件。把需编译的.NET EXE文件和对应的DLL文件复制到你在cygwin的工作文件夹中,如果你的windows用户名是xyz,那么这个文件夹就是 c:\cygwin\home\xyz\,(这一步不是必须的,如果你不怕麻烦而愿意多打字的话)。

2,转换与打包。通过下面的命令,将.net程序和类库打包并得到一个c程序源码(假设你需要转换的.NET文件是a.exe)。
    mkbundle -c -o b.c -oo b.o a.exe -z
    或者:
    mkbundle -c -o b.c -oo b.o a.exe aa.dll c:\\mono\\lib\\mono\\4.5\\mscorlib.dll -z
    或者:
    mkbundle -c -o b.c -oo b.o --dept a.exe -z

3,修改得到的c文件:
    这是本文的精华所在。
    为什么要修改这个c文件,很简单:
    A,不希望与exe文件相关的类库全部打包到一个文件中,否则,太浪费,而且影响启动速度。
    B,这个c文件是目标程序的关键文件,我希望在中间加上自己的东西,让我的程序如虎添翼。
    C,我程序要在中文、日文这样的含有非英文字母的文件夹中运行。

3.1,需要添加和修改的内容:
    A,用VS或记事本打开b.c,把下面的代码复制到main函数之前,作一个准备。

#include <dir.h>
    #include "/usr/include/iconv.h"
    int gbk_utf8(char *inbuf,int inlen,char *outbuf,int outlen){
        iconv_t cd;
        char **pin = &inbuf;
        char **pout = &outbuf;
        cd = iconv_open("utf-8","gbk");
        if (cd == 0) return -1;
        memset(outbuf, 0, outlen);
        if (iconv(cd, pin, &inlen, pout, &outlen) == -1) return -1;
        iconv_close(cd);
        return 0;
    }

B、在main函数中,找到下面这两行并注释或删除掉:
   if (config_dir != NULL && getenv ("MONO_CFG_DIR") == NULL)
      mono_set_dirs (getenv ("MONO_PATH"), config_dir);
   
   C、接着,就在这行下边,即“mono_mkbundle_init”一行之前,输入下边的代码:

const char* lib = "\\lib";
    const char* etc = "\\etc";

char p[strlen(argv[0])];
    wsprintf(p,"%s",argv[0]);

int l = 0;
    l = strlen(p);
    for(i=l-1; i>0; i--){
        if(p[i] == '\\'){
            p[i] = '\0';
            break;
        }
    }

l = strlen(p) + strlen(lib);
    char s_lib[l];
    wsprintf(s_lib, "%s%s", p, lib);

l = strlen(s_lib);
    char* s_lib_utf8 = (char*)malloc(l*2);
    memset(s_lib_utf8, 0, l*2);
    gbk_utf8(s_lib, l, s_lib_utf8, l*2);

l = strlen(p) + strlen(etc);
    char s_etc[l];
    wsprintf(s_etc, "%s%s", p, etc);

l = strlen(s_etc);
    char* s_etc_utf8 = (char*)malloc(l*2);
    memset(s_etc_utf8, 0, l*2);
    gbk_utf8(s_etc, l, s_etc_utf8, l*2);

mono_set_dirs(s_lib_utf8, s_etc_utf8);

接着在mono_mkbundle_init一行之后加入一行:
    chdir("c:\\");

最后,找到下面三行
    #ifdef _WIN32
    #include <windows.h>
    #endif
    并在“#endif”后加入一行:
    #undef _WIN32

改完了,存盘退出。
    (有人会说“输入这么多,为什么不写个函数以方便我将来复用?”,我说,这不是我的事。)

3.2,编译:
    用下面这个命令生成你的目标文件“b.exe”。
    gcc -mno-cygwin -o b.exe -Wall b.c `pkg-config --cflags --libs mono-2|dos2unix` b.o -lz -liconv

四,程序、类库、配置文件的组织:
    这一步,是为你的程序安一个家,让它真的能跑起来。

1,在某个盘,比如D盘,建个文件夹,比如是“myapp”
    把刚才编译得到的目标文件b.exe复制到D:\myapp文件夹中。
    同时把c:\mono\bin\文件夹中的mono-2.0.dll、zlib1.dll、iconv.dll复制到d:\myapp中。

2,组织类库
    在“d:\myapp”文件夹中,建lib和etc两个子目录。
    在lib文件夹中,建名叫“mono”的文件夹。
    在d:\myapp\lib\mono文件夹中,根据你.NET程序集版本号建一个文件夹,名字就是版本号,比如“4.5”,当然,你也可以把2.0、4.0也建好。
    如果你没有将mscorlib.dll打包到.EXE中,请将c:\mono\lib\mono\4.5\mscorlib.dll,复制到 d:\myapp\lib\mono\4.5这个文件夹中。
    在d:\myall\lib\mono文件夹中,建一个名为gac的文件夹,这个文件夹是用来放你的程序需要的mono版.NET类库的。
    放些什么?放你的exe、dll文件中引用到的那些程序集的库文件(如果你已经把这些文件打包到了.exe中,那么你就不需要放任何文件)。
    比如,你引用了System名字空间,那么,将c:\mono\lib\mono\gac文件夹下的System文件夹复制到D:\myapp\lib\mono\gac中就行了。

3,组织配置文件
    把c:\mono\etc文件夹中的“mono”文件夹复制到d:\myapp\etc文件夹中。
    用写字版打开config文件,找到并删除下列三行:
    <dllmap dll="gdiplus" target="/tmp/install/lib/libgdiplus.so" />
    <dllmap dll="gdiplus.dll" target="/tmp/install/lib/libgdiplus.so" />
    <dllmap dll="gtkhtml-3.0" target="libgtkhtml-3.8-15.dll"/>

通过上面的几个步骤,你的程序已经变成了可以独立运行的程序了,你把d:\myapp这个文件夹压缩打包,然后解压到没有安装.net的电脑上,试试。