C语言边角料5:一个跨平台的头文件

一、前言

我们平常在写代码的时候,特别是在制造轮子的时候(为别人提供库文件),会遇到各种不同的需求场景:

  1. 有些人需要在 Linux 系统下使用,有些人需要在 Windows 系统下使用;
  2. 有些人使用 C 语言开发,有些人使用 C++ 来开发;
  3. 有些人使用动态库,有些人使用静态库;

特别是在 Windows 系统中,库文件中导出的函数需要使用 _declspec(dllexport) 来声明函数,而使用者在导入的时候,需要使用 _declspec(dllimport) 来声明函数,甚是麻烦!

这篇短文分享一个头文件,利用这个头文件,再加上几个编译期间传递的宏,就可以完美的处理刚才所说的各种需求。

二、头文件

先直接上代码,可以先试着分析一下,后面我们再逐一分析不同的使用场景。

这个头文件的主要目的,就是定义一个宏:MY_API,然后把这个宏添加在库文件中每一个需要导出的函数或者类的声明中即可。例如:

 
 
 
 
  1. void MY_API do_work();

下面是头文件:

 
 
 
 
  1. _Pragma("once")
  2. #if defined(WIN32) || defined(_WIN32) || defined(__WIN32__)
  3.     #define MY_WIN32
  4. #elif defined(linux) || defined(__linux) || defined(__linux__)
  5.     #define MY_LINUX
  6. #endif
  7. #if defined(MY_WIN32)
  8.     #ifdef MY_API_STATIC
  9.       #ifdef __cplusplus
  10.          #define MY_API extern "C"
  11.       #else
  12.          #define MY_API
  13.       #endif
  14.    #else
  15.       #ifdef MY_API_EXPORTS
  16.          #ifdef __cplusplus
  17.             #define MY_API extern "C" __declspec(dllexport)
  18.          #else
  19.             #define MY_API __declspec(dllexport)
  20.          #endif
  21.       #else
  22.          #ifdef __cplusplus
  23.             #define MY_API extern "C" __declspec(dllimport)
  24.          #else
  25.             #define MY_API __declspec(dllimport)
  26.          #endif
  27.       #endif
  28.    #endif
  29. #elif defined(MY_LINUX)
  30.     #ifdef __cplusplus
  31.        #define MY_API extern "C"
  32.     #else
  33.        #define MY_API
  34.     #endif
  35. #endif

三、预定义的宏

假设需要写一个库文件,提供给别人使用。定义了上面这个头文件之后,其他的文件中都要include 这个头文件。

1. 平台宏定义

不同的平台预定义了相应的宏定义,例如:

  • Windows 平台:WIN32, _WIN32, WIN32;
  • Linux 平台:linux, __linux, linux;

在一个确定的平台上,这些宏不一定全部定义,很可能只有其中的某一个宏是被定义的。

为了统一性,我们在头文件的刚开始部分,把这些可能的宏统一起来,定义我们出我们自己的平台宏定义:MY_WIN32 或者是 MY_LINUX,后面需要区分不同的平台时,就用这个自己定义的平台宏。

当然,还可以继续扩充出其他平台,例如:MY_MAC, MY_ARM 等等。

2. 编译器宏定义

如果在写库代码的时候,使用的是 C++,而使用者使用的是 C 语言,那么就需要对库函数进行extern “C” 声明,让编译器不要对函数的名称进行改写。

编译器 g++ 预定义了宏 __cplusplus,因此,在头文件中,就利用了这个宏,在 MY_API 中添加 extern "C" 声明。

四、Windows 平台场景分析

1. 编译生成库文件

(1) 生成静态库

在静态库中,不需要 __declspec(dllexport/dllimport) 的声明,因此只需要区分编译器即可(gcc or g++),在编译选项中定义宏 MY_API_STATIC,即可得到最终的 MY_API 为:

  • gcc 编译器:#define MY_API
  • g++ 编译器:#define MY_API extern "C"

(2) 生成动态库

在编译选项中,定义宏 MY_API_EXPORTS,这样最终得到的 MY_API 就会变成:

  • gcc 编译器:#define MY_API __declspec(dllexport)
  • g++ 编译器:#define MY_API extern "C" __declspec(dllexport)

2. 使用库

在使用库的应用程序中,也需要在代码中 include 这个头文件,然后加上编译选项中定义的各种宏,来生成对应的 MY_API 宏定义。

(1) 使用静态库

需要在编译选项中定义 MY_API_STATIC,即可得到最终的 MY_API 为:

  • gcc 编译器:#define MY_API
  • g++ 编译器:#define MY_API extern "C"

(2) 使用动态库

在编译选项中不需要任何宏定义,即可得到最终的 MY_API 为:

  • gcc 编译器:#define MY_API extern "C" __declspec(dllimport)
  • g++ 编译器:#define MY_API __declspec(dllimport)

这样就相当于声明导入库函数了。

五、Linux 平台场景分析

Linux 平台下就简单多了,只需要注意编译器的问题,而没有导出和导入之分。

网站标题:C语言边角料5:一个跨平台的头文件
文章分享:http://www.shufengxianlan.com/qtweb/news9/453159.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联