`
liangguanhui
  • 浏览: 112955 次
  • 性别: Icon_minigender_1
社区版块
存档分类
最新评论

[原创] 如何使用VC2005编译“真正”的静态Qt应用程序

阅读更多

[原创] 如何使用VC2005编译“真正”的静态Qt应用程序

(我非常讨厌引用或者转载别人的文章的时候不注明地址或者作者,希望读者能够改变我这个观点  :)  —— 夏威夷雪人)

(下文只涉及windows,不谈及linux、unix等,另外,这里说的VC2005,是因为我用的是2005,其实其他版本的VC亦同样可以参考)

首先,你应该该知道什么叫静态引用编译、什么叫动态引用编译。我这里只是简单的提提,具体的可以google一下。

动态引用编译,是指相关的库,以dll的形式引用库。动态编译的Exe程序尺寸比较小,因为相关的库都没有包含进来。当然,程序发布的时候,还要把相关的库也一并发布出去。

静态引用编译,是指把相关的库也一并引入Exe文件。这是程序的尺寸就会很大,不过,程序发布就会变得简单很多。

其次,你可能会注意到我标题上写了“真正”这两个字。为什么我要强调真正这两个字呢?因为使用VC编译的C或者C++程序,都需要相关的C runtime库才能运行。如果你是VC6,相应的库就叫MSVCR,如果是vc2005,那就是MSVCR08,vc2008就是MSVCR09。我这里假设你安装的是VC2005,请进入如下目录:${VS Install Dir}\VC\redist\x86 和 ${System Driver}:\windows\WinSxS,你就会发现下面有很多很多的库。没错,这里相当一部分就是C runtime库。

好了,言归正传,首先,我们用VC2005写了一个不使用MFC的存C或者C++的程序,怎么发布给最终用户呢?有两个方法:
(1)静态引用C runtime库:打开“项目”->“XXX属性”->“配置属性”->“C/C++”->“代码生成”->“运行时库”。看到了吧?这里一共有四个选项,其中MT开头的是静态引用,MD开头的是动态引用,d结尾的是Debug调试版本,没有d的是Release发布版本,所以就一共有四个选项。我们选择/MT,然后编译程序(生成的程序应该不小),把这个程序发给用户,然后用户就可以直接运行了。
(2)动态引用C runtime库:跟上面差不多,不过是用/MD选项编译(程序应该只是几十K),然后发给用户。这时,用户是不能运行这个程序的,会报个什么程序引导失败,重装系统可能会修复问题之类的提示。这是我们还要把C runtime库一并发过去。把${VS Install Dir}\VC\redist\x86\Microsoft.VC80.CRT下的所有文件(注意,是所有,包括那个.manifest文件)发给用户,用户把这些文件放在我们的程序的同一个目录,然后再次运行,这时,程序就起来了(VC2005之后,C runtime库的引用改变了很多,建议google一下)。

说完C runtime库,就来说说Qt库了,这里我假设你用的是最新的Qt4.4.3。我们编译Qt的时候,configure.exe有很多参数,大家可以configure.exe --help来看看,其中,默认生成的Qt库(这里默认的意思,是指没有加-share或者-static参数)是动态引用的,也就是说,编译完后,在QtDir的lib目录下除了一大堆lib文件外,还有一大堆的dll文件。我们发布我们的Exe程序的时候,需要把相应的Qt库的Dll也一并发给用户。

按照Qt的安装手册和网上一大堆大牛的说法,加上-static参数后,Qt就可以静态编译了,也就是说,lib目录下之后一大堆lib文件,没有dll文件。是否?我们做个试验:

首先是设置变量:
set QTDIR=%CD%
set PATH=%PATH%;%QTDIR%\bin
set QMAKESPEC=win32-msvc2005
"C:\Program Files\Microsoft Visual Studio 8\VC\vcvarsall.bat"" x86
配置makefile:
configure -release -static -fast -qt-sql-odbc -qt-sql-sqlite -no-webkit
(这里的参数就不一一说明了,建议读者打入--help认真查查,特别注明一下,之所以-no-webkit,是因为新版的qt加上了Webkit,而这个东东编译的时候非常耗时间,编译后也很大,有100多M,并且我基本不会用到这个东东,所以忽略它)
然后
cd src  (我之所以直接进入src目录nmake,是因为不想make其它不相干的模块,节省时间)
nmake

漫长的等待之后,我们发现lib下果然只有一大堆lib文件了,而且每个lib文件的尺寸都在M以上,似乎已经成功了。然后我们在安装了qt-vsintegration的VC2005新建一个Qt工程,然后编译一个release版本。编译的时候,问题来了。我们选择/MD选项,这时链接就可以通过,但如果我们想要用/MT选项来使用静态C runtime库,就会报一大堆某某函数链接重复之类的错误。经验告诉我们,之所以不能使用/MT来编译,是因为另外一个库——Qt库使用了另外一种引用方式/MD(原则上来说,一个程序里面的所有模块,都应该使用同一种引用方式,具体可以google一下)。很显然,我们编译的所谓静态Qt程序,一样要背着微软的C Runtime库到处跑,还不够“真正”的静态。


怎么才能做成完全的静态呢?记得之前编译wxWidgets的时候,它除了有SHARED=0或者1的选项之外,还有一个RUNTIME_LIBS = static or dynamic的选项,很显然,这个RUNTIME_LIBS的选项就是我们想要的选项。不过我翻遍了Qt的安装手册以及网上大牛的文章,都没有提及这个问题,我当时心里就觉得奇怪,难道没人遇到过这个问题?我又认真翻查了configure.exe的help,也没有类似的选项,问题一下就僵住了。

回忆一下刚才我们编译的时候,屏幕上调用cl.exe编译的时候,有这样一个参数:cl.exe .... -MD .... xxx.cpp,眼利的朋友一下就会发现,这个-MD就是c runtime动态引用的选项。然而,怎么把这个-MD改成-MT呢?我们翻开刚才我们编译的qt的src目录下,随便找个目录进去,打开Makefile.Release,我们就会看到CFLAGS=-MD ........,没错,就是这里。我们只要在这里把-MD改成-MT,就会使用静态c runtime库编译Qt了。我们当然不可能一个一个地替换这些makefile,关键是找出生成这些参数的模板文件。很显然,它肯定在qt的mkspecs目录,我们直奔win32-msvc2005目录,果然找到一个qmake.conf文件,果然找到一个QMAKE_CFLAGS_RELEASE = -O2 -MD,把这里的-MD换成-MT,然后清理一下刚才的生成的配置信息(网上又说用nmake confclean来清空,不过我没有成功,貌似是使用了-fast参数的缘故,不过没关系,把这个目录删掉,重新解压一份源代码就可以了,然后把win32-msvc2005目录下的qmake.conf的-MD换成-MT),重新
configure -release -static -fast -qt-sql-odbc -qt-sql-sqlite -no-webkit
然后nmake
又是漫长的等待。不过我们不要干等,看看出来的编译命令,cl.exe .... -MT .... xxx.cpp,果然变成静态c运行库了。
编译完之后,像刚才那样,在VC2005建一个Qt的工程,然后用/MT这个选项编译,OK,编译成功,出来的Exe文件大小是4.95M,貌似已经把C runtime库嵌进来了。然后把这个程序放到用户那里运行,OK单个Exe文件运行成功了。

至此,编译真正静态的Qt程序试验完成。总结一下整个过程,首先是要有耐性,应为编译一次Qt都至少两个小时(当然,用一些技巧,例如-fast,-no-qmake,只编译src等等的技巧可以缩短很多时间),我来回就编译了五次Qt;其次熟悉一些常见的编译、链接的错误,例如一见到XXX库已经引用之类的错误,马上就联想到应该是引用不同的库导致的;最后,要善于发现问题,查找问题。

                       —— 夏威夷雪人(liangguanhui@163.com)
                            2008年11月11日 光棍节 21:18

分享到:
评论

相关推荐

    staticlib5.6.3静态编译库

    这是我自己编译的可用于windows 32位系统下的QT静态编译库,此库为官方源码完全编译...由于自己编译静态库对初学者有一定难度,所以我在这里提供了已经编译好的完整静态编译库与编译库的使用教程,适合初学者直接使用。

    Qt极限精简静态编译,生成的EXE只有2.2M(未压缩)

    本文将详细解析如何进行Qt的极限静态编译,以及使用VC6编译器生成的小巧精悍的EXE文件。 首先,我们要理解静态编译的概念。静态编译意味着在构建应用程序时,所有依赖的库都会被整合到最终的可执行文件(EXE)中,...

    QT4.3.3+VC6.0安装编译教程

    如果仅用于程序发布,可以通过以下命令仅编译静态库: - 使用MinGW编译器: ``` mingw32-make release sub-src ``` - 使用VC2005编译器: ``` nmake release sub-src ``` - **注意事项**:此过程耗时较长,...

    QT 5.14.2/windows msvc x64静态编译好的目录打包

    QT 5.14.2 静态编译好的目录打包,包含所有的lib、bin、include、plugin目录等。 编译真的很麻烦,我打包好了,需要的拿去吧! 使用方法:把目录解压,在Qt vs tools里面配置一下路径就可以使用了 注: MTd方式,...

    vs2008 md方式静态编译qt4.8.6库

    windows平台下,vs2008使用md的编译方式,编译的qt4.8.6版本的静态库。 压缩包中包含了编译后的二...使用qt开发的程序不想带上qt的dll时,可以使用压缩包中的库进行开发,由于是md编译的,程序需要带上vc的运行时库。

    Mupdf1.13.0静态库;VC2017 x64编译,亲测Qt可调用;

    总结来说,Mupdf1.13.0静态库为开发者提供了一种简单的方式,在使用Visual Studio 2017和Qt5.14的64位环境中,轻松地在应用程序中集成PDF阅读功能。正确理解和使用这个库,可以提升用户体验,同时简化PDF相关的编程...

    Qt源码静态编译.docx

    在默认情况下,使用Qt Creator编译的Qt应用程序是动态链接的,这意味着生成的exe文件需要依赖于Qt的dll文件才能运行。例如,一个简单的点击计数应用,其编译后的exe文件大小只有25KB,但需要47MB的dll文件才能正常...

    exiv2静态编译

    在这个场景中,我们关注的是如何将Exiv2静态编译为适用于Qt或Visual C++ (VC)项目的静态链接库。 静态编译Exiv2的主要目的是避免在程序运行时依赖于外部的DLL文件。动态链接库(DLL)虽然可以节省磁盘空间,但需要...

    Qt5.13静态编译及VS2019开发环境配置.pdf

    最新版Qt5.13静态库编译方法详细图文说明, 以及演示在VS2019下配置和使用Qt5.13静态库的方法。使用Qt5.13静态库编译后的可执行程序,不需依赖大量的dll文件,一个执行文件即可到处运行。

    VS2017+QT如何创建和使用静态库.rar

    VS2017+QT如何创建和使用静态库的Demo Code,简单的在QT项目中使用VC的静态库,里面包括了如何创建和使用。 博客地址:https://blog.csdn.net/ZHONGCAI0901/article/details/106886208

    适合window 64 vs2015 qt5.8 静态编译对应的openssl编译64位的版本

    适合window 64 vs2015 qt5.8 静态编译对应的openssl编译64位的版本,openssl-1.0.2i,目前已经集成到我的 qt项目中,而且试过openssl可以用 文章可以参看:...

    qt_static_32.7z

    使用VC2008编译的Qt静态库与该编译器版本兼容,意味着你的项目必须使用相同或兼容的编译器来避免潜在的链接错误。 **压缩包内容分析** - **q3porting.xml**:可能包含Qt从旧版本到新版本的迁移指南和建议,帮助...

    QT中pro文件使用的系统变量的说明

    例如,CONFIG += qt warn_on release 指定应用程序使用 Qt 库,输出警告信息,并生成一个发布的应用程序。CONFIG 系统变量可以指定以下选项: * qt:应用程序使用 Qt 库 * thread:应用程序是一个多线程应用程序 * ...

    Qt 5.11.2 MSVC2017 32位静态编译版本.txt

    Qt 5.11.2 MSVC2017 32位静态编译版本, Qt 5.11.2 MSVC2017 32位静态编译版本 编译了好几次才成功

    QT 5.15.0/windows msvc X64静态编译好的目录打包

    QT 5.15.0 静态编译好的目录打包,包含所有的lib、bin、include、plugin目录等...完全静态编译方式:MT(多线程),不需要任何外部运行库,包括QT和VC运行库 64位版本,支持VS2017和VS2019,QWebEngine不是静态编译的

    Qt5.15.0_Static_msvc2019_64_MT多线程.rar

    QT 5.15.0 静态编译好的目录打包,包含所有的lib、bin、include、plugin目录等...完全静态编译方式:MT(多线程),不需要任何外部运行库,包括QT和VC运行库 64位版本,支持VS2017和VS2019,QWebEngine不是静态编译的

    Qt动态库 导出类

    7. **发布应用程序**:如果使用Qt动态库,发布应用程序时,除了可执行文件,还需要包含所有必要的Qt动态库。在Windows上,这通常意味着打包msvcrXX.dll等运行时库;在Linux上,可能需要提供libstdc++.so等。 8. **...

    coin3D-3.1.3-SoQt-SoWin-VC10.7z )

    在Windows系统中,开发者经常使用Coin3D结合Qt或SoWin来创建具有高效3D图形功能的应用程序。 1. **Coin3D**: Coin3D是基于Open Inventor的API实现,它提供了一个高级别的、面向对象的C++接口,用于处理3D图形渲染和...

Global site tag (gtag.js) - Google Analytics