通常来说,在大项目里,用动态链接比静态链接体积要小,缺点就是要带上一个crt库。windows现在又引入了manifest,可以指定crt的版本号,这样对于向上兼容是个不错的方法。
如果作为单独的库发布,那么带上crt可能不太现实,因为它们确实不小。所以需要用静态链接。gcc静态连接很方便,只需要把相应的.a文件加入即可。vc2010需要将/MDd改为/MTd。虽然是这样,但如果使用了别人的库,可能它本身是/MDd的,这就比较棘手了,这就是为什么有很多软件要带上msvcp100.dll、msvcr100.dll等的原因了。这两个文件总共约1M左右。
/MTd还有能遇到的问题是LINK2001等符号重定义的问题,下面是常见的几个会重定义的lib及其说明
LIBC.LIB Single-threaded, static link /ML
LIBCMT.LIB Multithreaded, static link /MT _MT
MSVCRT.LIB Multithreaded, dynamic link (import library for MSVCR71.DLL). Be aware that if you use the Standard C++ Library, your program will need MSVCP71.DLL to run. /MD _MT, _DLL
这里编译器会提示你用/NODEFAULTLIB:来忽略特定的库。运气好的时候能work,但通常是忽略了某个后影响另一个。目前有两种方法解决
1. 找出有依赖问题的文件,单独编译(我没试过)
2. 启用/FORCE:MULTIPLE,有一定副作用
Boost静态连接
Boost中大部分库只需包含头文件即可,除了下面几个
- Boost.Filesystem
- Boost.GraphParallel
- Boost.IOStreams
- Boost.MPI
- Boost.ProgramOptions
- Boost.Python (see the Boost.Python build documentation before building and installing it)
- Boost.Regex
- Boost.Serialization
- Boost.Signals
- Boost.System
- Boost.Thread
- Boost.Wave
bjam link=static runtime-link=static
VS会自动连接到相应的静态库。
PS: 有些库即使静态连接,Debug版本也会依赖msvcp和msvcr(主要是些调试函数),编译为Release就没有了。