[原创]#pragma 指示符应用举例

 

    尽管 C 和 C++ 都已经有标准,但是几乎每个编译器 (广义,包含连接器等) 扩展一些 C/C++ 关键字。合理地应用这些关键字,有时候能使我们的工作非常方便。下面随便说说 Visual C++ 中 #pragma 指示符的使用。

一、用#pragma导出DLL函数

    传统的到出 DLL 函数的方法是使用模块定义文件 (.def),Visual C++ 提供了更简洁方便的方法,那就是“__declspec()”关键字后面跟“dllexport”,告诉连接去要导出这个函数,例如:

__declspec(dllexport) int __stdcall MyExportFunction(int iTest);

    把“__declspec(dllexport)”放在函数声明的最前面,连接生成的 DLL 就会导出函数“_MyExportFunction@4”。

    上面的导出函数的名称也许不是我的希望的,我们希望导出的是原版的“MyExportFunction”。还好,VC 提供了一个预处理指示符“#pragma”来指定连接选项 (不仅仅是这一个功能,还有很多指示功能) ,如下:

#pragma comment(linker,"/EXPORT:MyExportFunction=_MyExportFunction@4")

    这下就天如人愿了:)。如果你想指定导出的顺序,或者只将函数导出为序号,没有 Entryname,这个预处理指示符 (确切地说是连接器) 都能够实现,看看 MSDN 的语法说明:

/EXPORT:entryname[,@ordinal[,NONAME]][,DATA]

   @ordinal 指定顺序;NONAME 指定只将函数导出为序号;DATA 关键字指定导出项为数据项。

二、指示文件只包含一次

    在头文件中,一般在整个工程中我们只要包含一次就够了,但是如果我在多个 .c/.cpp 文件中都要包含着个头文件,比如 Windows.h,那很多声明等等岂不是有两次了?解决这个问题的传统的方法是在头文件开始出用 #define 定义一个宏,比如 Windows.h 中:

#ifndef _WINDOWS_
#define _WINDOWS_

</< p="" />

 

   然后在文件结为加上 #endif,这样就可以避免被包含多次。但是这样的后果是代码的可读性较差 (个人观点),VC 给我们提供了另外一个途径,那就是在文件的前面加上:

#pragma once

    是不是很方便?

三、使警告无效

    有时候我们不得不对变量进行强制转换,由此引来编译器的一番警告,特别是 C++ 中,类型检查相对于 C 更为严格。这虽然不影响什么,但是看起来多不爽——我是故意要这样的,你警告什么!:)这时候你看到警告类型,比如“warning C4311: “类型转换” : 从“HHOOK”到“BOOL”的指针截断”,在前面加上:

#pragma warning(disable: 4311)

 

   编译器就没话说了:)。

四、指定连接要使用的库

    比如我们连接的时候用到了 WSock32.lib,你当然可以不辞辛苦地把它加入到你的工程中。但是我觉得更方便的方法是使用 #pragma 指示符,指定要连接的库:

#pragma comment(lib, "WSock32.lib")

五、显示编译消息


    没多少用处,举个例子吧:

#ifdef _DEBUG
#pragma message("编译连接为调试模式...")
#endif // _DEBUG

 

   其实“#pragma”的功能远远不止于此,以上只是冰山一角而已。具体的使用方法,详细情况清参看 MSDN,这里仅仅是说说有这些功能而已:)

### `#pragma once` 的用法及示例 在 C 语言中,`#pragma once` 是一种预处理指令,用于防止头文件在同一个编译单元中被多次包含,从而避免重复定义的错误。与传统的 `#ifndef`、`#define`、`#endif` 相比,`#pragma once` 更加简洁,且在某些编译器中执行效率更高[^3]。 #### 用法说明 `#pragma once` 应该放置在头文件的最开始处,确保预处理器在第一次处理该头文件后,不再重复处理。编译器会根据文件的唯一标识(如 inode 或路径)来判断是否已经包含过该头文件。 #### 示例说明 以下是一个使用 `#pragma once` 的头文件示例: ```c // myheader.h #pragma once void print_hello(); ``` 在对应的源文件中,可以正常引用该头文件而无需担心重复包含问题: ```c // main.c #include "myheader.h" #include <stdio.h> void print_hello() { printf("Hello, world!\n"); } int main() { print_hello(); return 0; } ``` 上述示例中,`myheader.h` 使用了 `#pragma once` 来防止重复包含,简化了头文件的保护机制[^1]。 #### 与传统方式的区别 传统的头文件保护方式依赖于宏定义,如下所示: ```c // myheader.h #ifndef MYHEADER_H #define MYHEADER_H void print_hello(); #endif // MYHEADER_H ``` 这种方式虽然兼容性更好,但需要手动定义宏名,并且在嵌套包含时仍然会多次执行预处理指令。而 `#pragma once` 则直接由预处理器控制,确保只执行一次,效率更高[^3]。 #### 编译器支持情况 尽管 `#pragma once` 提供了更高效的头文件保护机制,但其并非 C 语言标准的一部分,因此不同编译器的支持情况可能不同。例如,Microsoft Visual C++ 和 GCC 等主流编译器都支持该指令,但在一些老旧或特定平台的编译器中可能不被支持[^2]。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值