Windows 平台下构建体系大致分三类:原生 Visual C++、MinGW、Cygwin。下面详细介绍着三种类型:
Visual C/C++
VisualStudio
微软的原生 C/C++ 编译器实现,支持标准 C 和 C++,当然在标准 C 和 C++ 基础上有大量扩展。另外,宇宙第一 IDEVisual Studio
提供了编辑、编译、调试一体的解决方案,更是给编码提供了很多的效率提升。随着微软.NET
战略的实施,VS 逐渐成为. NET 开发者的必要开发工具。
MinGW
MinGW
MinGW
(Minimal GUN for Windows) 是一款 windows 软件,旨在将 GNU 编译工具链 (gcc
,binutil
) 移植到 windows 下,这样可以在 windows 下使用 gcc 等工具构建软件。
一般 MinGW 需要配合MSYS
来用。MSYS
是一个 Shell 的 windows 移植版本,附带有一些常用的命令和工具。这样,有了MSYS
,连configure
都可以用了。MSYS
实际上是下面要提到的Cygwin
的一个早期分支派生出来的。
MSYS2
是重写了MSYS
,并引入了pacman
包管理工具。用MSYS2
可以比较方便的搭建 MinGW 开发环境。
不过 MinGW 环境只是一系列编译构建工具的集合。对于在 windows 下开发程序,仍然需要使用 windows 原生的 api。换句话说,可以理解为用 MinGW 代替 Visual C/C++。
Cygwin
cygwin
Cygwin
在 windows 上实现了一层posix
层,以便基于 posix 的程序移植到 windows。因此,基于Cygwin
,可以获得更多的 linux 程序,并在 windows 使用。由于采用了中间层,因此运行效率相对原生的 windows 有些低,但是如果希望快速将 linux 程序移植到 windows,是一个不错的选择。
Cygwin
也派生出了上面提到的MSYS
。
在发布基于Cygwin
的程序时,需要带上cygwin1.dll
,作为依赖。
基于Cygwin
我们可以在 windows 上使用熟悉的git
vim
make
gcc
等软件,默认的终端可以让我们继续使用/
而不是\
作为路径分隔符。
Denpendency Walker
: 在 linux 下我们用ldd
来查看程序的依赖,在 windows 下,我们需要 Denpendency WalkerDumpbin
: VS 自带的工具,用于查看库的导入导出函数MSDN
:Man
是 linux 最重要的文档参考工具,而在 windows 下,MSDN 是必备的《Windows via C/C++》
:此书的地位相当于APUE
VS Command Line Tool
:作为习惯了终端界面的 linux 程序员来说,可以更愿意使用编辑器 + 编译器的模式,而并不喜欢使用 vs 的集成环境界面。那么我们需要找到 VS 附带的 Visual Studio 命令行工具,用管理员身份打开它。这个 bat 脚本会为我们设置必要的编译环境变量,然后在其中运行cl
,link
,nmake
等编译构建命令才能有作用。
windows 中也有类似glibc
的 C 运行库,以支撑 C/C++ 编写的程序得以运行。他们是 MSVCRTxxd.dll 和 MSVCRTxx.dll,xx 是版本号,笔者见过的有 MSVCRT70.dll MSVCRT80.dll MSVCRT90.dll MSVCRT100.dll 等,末尾的d
表示是 debug 版本的运行库还是 release 版本的运行库。程序运行的时候究竟依赖哪个版本,取决于在编译阶段采用/MD
还是/MDd
编译选项。
在 windows 中库文件分两种:
- .obj:对象文件,相当于
.o
。 - .dll:动态链接库文件,类似
.so
。程序在运行时才会加载和链接需要的 dll。windows 的一个方便的地方是,dll 只需要在你的 exe 文件所在目录中即可。 - .lib:静态库文件,类似
.a
。程序在编译时如果选择静态链接,那么. lib 会用上。不过并不是所有的. lib 都是静态库文件,还有一些. lib 比较小,是作为. dll 的导入库存,没有实际的二进制代码,只是一些符号表。也就是说如果你的程序依赖一个. dll,那么在编译阶段,编译器要找到. dll 相应. lib 导入库才能完成链接。而在运行时,只需要找到 dll 即可。
在 windows 中,可执行文件采用 PE 格式封装,相应的,linux 中采用 ELF。
编写或使用动态链接库需要注意函数声明方式。VC 编译器有一组特殊的声明:__declspec(dllexport)
和__declspec(dllimport)
分别用于在 dll 中声明为导出函数,和在使用 dll 时声明为 dll 导入函数。这种声明会将函数名字添加__impl_
前缀,因此如果不能配对使用的话,往往容易出现链接错误。
此外,__stdcall
和__cdecl
影响函数的入参顺序。
https://www.jianshu.com/p/edbbf71fe22a