从Linux到Windows必知 - 图1

Windows 平台下构建体系大致分三类:原生 Visual C++、MinGW、Cygwin。下面详细介绍着三种类型:

Visual C/C++

从Linux到Windows必知 - 图2

VisualStudio

微软的原生 C/C++ 编译器实现,支持标准 C 和 C++,当然在标准 C 和 C++ 基础上有大量扩展。另外,宇宙第一 IDEVisual Studio提供了编辑、编译、调试一体的解决方案,更是给编码提供了很多的效率提升。随着微软.NET战略的实施,VS 逐渐成为. NET 开发者的必要开发工具。

MinGW

从Linux到Windows必知 - 图3

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

从Linux到Windows必知 - 图4

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 Walker
  • Dumpbin: VS 自带的工具,用于查看库的导入导出函数
  • MSDN: Man是 linux 最重要的文档参考工具,而在 windows 下,MSDN 是必备的
  • 《Windows via C/C++》:此书的地位相当于APUE
  • VS Command Line Tool:作为习惯了终端界面的 linux 程序员来说,可以更愿意使用编辑器 + 编译器的模式,而并不喜欢使用 vs 的集成环境界面。那么我们需要找到 VS 附带的 Visual Studio 命令行工具,用管理员身份打开它。这个 bat 脚本会为我们设置必要的编译环境变量,然后在其中运行cllinknmake等编译构建命令才能有作用。

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