返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C语言详细讲解#pragma的使用方法
  • 619
分享到

C语言详细讲解#pragma的使用方法

2024-04-02 19:04:59 619人浏览 泡泡鱼
摘要

目录一、#pragma 简介二、#pragma message三、#pragma once四、#pragma pack五、小结一、#pragma 简介 #pragma 用于指示编译器

一、#pragma 简介

#pragma 用于指示编译器完成一些特定的动作

#pragma 所定义的很多指示字是编译器特有的

#pragma 在不同的编译器间是不可移植的

  • 预处理器将忽略它不认识的 #pragma 指令
  • 不同的编译器可能以不同的方式解释同一条 #pragma 指令

一般用法:

#pragma parameter

注:不同的 parameter 参数语法和意义各不相同

二、#pragma message

  • message 参数在大多数的编译器中都有相似的实现
  • message 参数在编译时输出消息到编译输出窗口中
  • message 用于条件编译中可提示代码的版本信息

例如:

#if defined(Android20)
    #pragma message("Compile Android SDK 2.0...")
    #define VERSioN "Android 2.0 "
#endif

注:#error 和 #warning 不同,#pragma message 仅仅代表一条编译消息,不代表程序错误。

下面看一个 #pragma message 使用示例:

test.c:

#include <stdio.h>
 
#if defined(ANDROID20)
    #pragma message("Compile Android SDK 2.0...")
    #define VERSION "Android 2.0"
#elif defined(ANDROID23)
    #pragma message("Compile Android SDK 2.3...")
    #define VERSION "Android 2.3"
#elif defined(ANDROID40)
    #pragma message("Compile Android SDK 4.0...")
    #define VERSION "Android 4.0"
#else
    #error Compile Version is not provided!
#endif
 
int main()
{
    printf("%s\n", VERSION);
 
    return 0;
}

下面为输出结果:

三、#pragma once

  • #pragma once 用于保证头文件只被编译一次
  • #pragma once 是编译器相关的,不一定被支持

下面两种方式的区别是:前者是被 C 语言所支持的,并不是只包含一次头文件,而是包含多次,然后通过宏控制是否嵌入到源代码中,也就是说通过宏的方式,可以保证头文件里面的内容只被嵌入一次,但是由于包含了多次,预处理器还是处理了多次,所以效率上来说比较低;后者是告诉预处理器当前文件只编译一次,所以说效率较高。

下面看一个 #pragma once 的使用示例:

global.h:

 #pragma once
 
int g_value = 1;

test.c:

#include <stdio.h>
#include "global.h"
#include "global.h"
 
int main()
{
    printf("g_value = %d\n", g_value);
 
    return 0;
}

下面为输出结果,可以看到虽然在test.c 定义了两次global.h,但是程序编译没有报错,且能正常运行,这就是 #pragma once 作用的结果:

工程应用中既想要移植性,又想要保证效率,可以采用以下做法:

global.h:

#ifndef _GLOBAL_H_
#define _GLOBAL_H_
 
#pragma once
 
int g_value = 1;
 
#endif

四、#pragma pack

什么是内存对齐?

  • 不同类型的数据在内存中按照一定的规则排列
  • 而不一定是顺序的一个接一个的排列

下面想想 Test1 和 Test2 所占的内存空间是否相同?

答案是否定的,Test1 占用 12 个字节,而 Test 占用 8 个字节

为什么需要内存对齐?

  • CPU对内存的读取不是连续的,而是分成块读取的,块的大小只能是1、2、4、8、16...字节
  • 当读取操作的数据未对齐,则需要两次总线周期来访问内存,因此性能会大打折扣
  • 某些硬件平台只能从规定的相对地址处读取特定类型的数据,否则产生硬件异常

#pragma pack 用于指定内存对齐方式

#pragma pack 能够改变编译器的默认对齐方式

例如,下面代码中,Test1 和 Test2 的内存均为 8 个字节

struct 占用的内存大小

第一个成员起始于 0 偏移处

每个成员按其类型大小和 pack 参数中较小的一个进行对齐

  • 偏移地址必须能被对齐参数整除
  • 结构体成员的大小取其内部长度最大的数据成员作为其大小

结构体总长度必须为所有对齐参数的整数倍

编译器在默认情况下按照 4 字节对齐。

下面通过代码,手工计算结构体所占用的内存大小

test.c:

#include <stdio.h>
 
#pragma pack(4)
struct Test1
{                   //对齐参数      偏移地址        大小
    char  c1;       //1            0              1
    short s;        //2            2              2
    char  c2;       //1            4              1
    int   i;        //4            8              4
};
#pragma pack()
 
#pragma pack(4)
struct Test2
{                   //对齐参数      偏移地址        大小
    char  c1;       //1            0              1
    char  c2;       //1            1              1
    short s;        //2            2              2
    int   i;        //4            4              4
};
#pragma pack()
 
int main()
{
    printf("sizeof(Test1) = %d\n", sizeof(struct Test1));
    printf("sizeof(Test2) = %d\n", sizeof(struct Test2));
 
    return 0;
}

以 Test1 为例,c1 类型大小为 1 字节,而 pack 参数默认为 4,所以对齐参数取最小为 1;同理 s 的对齐参数为 2,偏移地址要被 2 整除,所以为 2;同理 c 的对齐参数为 1,偏移地址为 4;同理,i 的对齐参数为 4,因为偏移地址要能被 对齐参数 4 整除,在偏移地址 4 之后能被 4 整除的最小偏移地址为 8,所以 i 的偏移地址为 8,而 i 占用 4 个字节,所以 Test1 结构体占用的总字节数为 12 字节,这就和上面的图对应起来了。

下面再通过一个例子感受一下:

GCc 编译器下,test.c:

#include <stdio.h>
 
#pragma pack(8)
 
struct S1
{                   //对齐参数      偏移地址        大小
    short a;        //2            0              2
    long b;         //4            4              4
};
 
struct S2
{                   //对齐参数      偏移地址        大小
    char c;         //1            0              1
    struct S1 d;    //4            4              8
    double e;       //4            12             8
};
 
#pragma pack()
 
int main()
{
    printf("%d\n", sizeof(struct S1));
    printf("%d\n", sizeof(struct S2));
 
    return 0;
}
 

下面为输出结果:

这里注意两点:

1.结构体成员的大小取其内部长度最大的数据成员作为其大小,所以 S1 的内存大小为 4,而pack 参数默认为 8,所以对齐参数为 4

2.一般的 pack 对齐格式分别是 1,2,4,8,16,默认的对齐格式,也就是:#pragmapack() 的情况下,会在结构体中挑选占用字节最多的类型,例如 double 占用 8 个字节

3.gcc 编译器暂时不支持 8 字节对齐,默认按照 4 字节对齐,所以 S2 中的 e 的对齐参数为 4,故 S2 占用内存大小为 20 字节。(学习采用的为 ubuntu 10.10)

如果把代码放在 VS2012 里运行,那结果就是和分析的一样,S2 占用 24 字节。

#include <stdio.h>
 
#pragma pack(8)
 
struct S1
{                   //对齐参数      偏移地址        大小
    short a;        //2            0              2
    long b;         //4            4              4
};
 
struct S2
{                   //对齐参数      偏移地址        大小
    char c;         //1            0              1
    struct S1 d;    //4            4              8
    double e;       //8            16             8
};
 
#pragma pack()
 
int main()
{
    printf("%d\n", sizeof(struct S1));
    printf("%d\n", sizeof(struct S2));
 
    return 0;
}
 

结果:

五、小结

#pragma 用于指示编译器完成一些特定的动作

#pragma 所定义的很多指示字是编译器特有的

  • #pragma message 用于自定义编译消息
  • #pragma once 用于保证头文件只被编译一次
  • #pragma pack 用于指定内存对齐方式

到此这篇关于C语言 详细讲解#pragma的使用方法的文章就介绍到这了,更多相关C语言 #pragma内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C语言详细讲解#pragma的使用方法

本文链接: https://lsjlt.com/news/146495.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C语言详细讲解#pragma的使用方法
    目录一、#pragma 简介二、#pragma message三、#pragma once四、#pragma pack五、小结一、#pragma 简介 #pragma 用于指示编译器...
    99+
    2024-04-02
  • C语言 详细讲解#pragma的使用方法
    #pragma是C语言中的预处理指令,用于告诉编译器在编译过程中采取特定的行为。以下是#pragma的使用方法的详细讲解:1. #p...
    99+
    2023-08-08
    C语言
  • C语言之pragma用法详解
    (0)前言 C语言Pragma 指令的作用是设定编译器的状态或者是指示编译器完成一些特定的动作。#pragma 指令对每个编译器给出了一个方法,在保持与C和C++语言完全兼容的情况下,给出主机或操作系统专有的特征。依据定义,编译指示是机器或...
    99+
    2023-10-12
    c语言 java linux c++ 开发语言
  • C语言详细讲解const的用法
    目录一、int const a / const int a二、const int(*p)/int const(*p)三、int*const p四、const用于函数的地址传递参数一、...
    99+
    2024-04-02
  • C语言详细讲解while语句的用法
    目录while语句格式例题1例题2例题3while语句格式 格式: while(表达式){    语句块} 1、先执行while(表达式),如条件为真执行语句块;...
    99+
    2024-04-02
  • C语言详细讲解qsort函数的使用
    目录qsort1.int型2.float型3.struct型qsort 功能:Performs a quick sort.(快速排序)参数:void qsort( void *bas...
    99+
    2024-04-02
  • C语言超详细讲解指针的使用
    目录指针概述自身类型指向类型代码例子数值型指针字符型指针单字符字符数组字符串型指针字符数组总结指针概述 C语言中指针也可以认为是一种类型,不同于数值型和字符型的类型。推演过去指针变量...
    99+
    2024-04-02
  • C语言详细分析讲解struct与union使用方法
    目录一、struct 的小秘密二、结构体与柔性数组三、C语言中的 union四、小结一、struct 的小秘密 C语言中的 struct 可以看作变量的集合 struct ...
    99+
    2024-04-02
  • C语言详细讲解注释符号的使用
    目录一、注释规则二、注释中一个有趣的问题三、教科书型注释四、迷惑型的注释五、忽悠型注释六、搞笑型注释七、漂亮的程序注释八、小结一、注释规则 编译器在编译过程中使用空格替换整个注释字符...
    99+
    2024-04-02
  • C语言详细讲解位运算符的使用
    目录一、位运算符分析二、小贴士三、位运算与逻辑运算四、小结一、位运算符分析 C语言中的位运算符 位运算符直接对 bit 位进行操作,其效率最高。 &按位与|按位或^按位异或~...
    99+
    2024-04-02
  • C语言超详细讲解getchar函数的使用
    目录一、getchar 函数二、缓冲区1、什么是缓冲区2、为什么要存在缓冲区3、缓冲区的类型4、缓冲区的刷新三、getchar 函数的正确使用1、getchar 的换行问题2、get...
    99+
    2024-04-02
  • C语言详细讲解指针数组的用法
    目录1. 指针数组定义方法2. 指针的指针(二级指针)3. 字符串和指针4. 数组指针定义方法数组指针的用法1. 指针数组定义方法 格式: 类型说明符 *数组名[ 元素个数 ] in...
    99+
    2024-04-02
  • C语言详细讲解二分查找用法
    目录【力扣题号】704.二分查找 力扣题目链接 示例 1: 输入: nums = [-1,0,3,5,9,12], target = 9     输出:...
    99+
    2024-04-02
  • C语言详细讲解if语句与switch语句的用法
    目录一、if 语句二、switch 语句三、错误提示一、if 语句 格式: if(写条件){输出内容}条件为真运行这个。 else {输出内容}否则输出这个。 代码: #includ...
    99+
    2024-04-02
  • C++BoostLockfree超详细讲解使用方法
    目录一、说明二、示例和代码Boost.Lockfree 一、说明 Boost.Lockfree 提供线程安全和无锁容器。可以从多个线程访问此库中的容器,而无需同步访问。 在 1.56...
    99+
    2022-11-21
    C++ Boost Lockfree C++ Lockfree方案
  • c语言pragma的用法是什么
    `#pragma` 是一个编译指令,用于告诉编译器执行特定的操作或者设置特定的编译选项。在C语言中,`#pragma` 可以用于以下...
    99+
    2023-09-04
    c语言
  • C语言详细讲解逻辑运算符的使用
    目录一、&& 与 II 分析二、!分析三、小结一、&& 与 II 分析 下面的程序运行结束后,i, j,k 的值分别为多少? #include <...
    99+
    2024-04-02
  • C语言超详细讲解宏与指针的使用
    目录1、关于define2、初识指针(1)内存(2)示例(3)指针的使用示例(4)指针变量的大小1、关于define define是一个预处理指令,有两种用法,一种是用define定...
    99+
    2024-04-02
  • C语言详细讲解循环语句的妙用
    目录一、循环语句分析二、do ... while 语句的循环方式三、while 语句的循环方式四、for 语句的循环方式五、break和 continue 的区别六、do 和 bre...
    99+
    2024-04-02
  • C语言详细讲解#error与#line如何使用
    目录一、#error 的用法二、#line 的用法三、小结一、#error 的用法 #error 用于生成一个编译错误消息 用法 #error message,message不需要用...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作