GNU C编译器扩展关键字:__attribute__
目录
一、section
二、aligned
三、packed
四、format
五、weak
六、alias
七、noinline和always_inline
GNU C增加了一个__attribute__关键字用来声明一个函数、变量或类型的特殊属性,可以知道编译器在编译过程中进行特定方面的优化或代码检查。
目前,__attribute__属性支持十几种属性声明:
section、aligned、packed、format、weak.......
一、section
section属性的作用是在程序编译时,将一个函数或变量放到指定的段,即section中。
一个可执行文件主要由代码段、数据段、BSS段构成。除了这三个段之外,可执行文件还包含其他一些段,如只读数据段,符号表等。在Linux系统中,可以使用readelf -S命令查看一个可执行文件的各个端信息,包括大小、起始地址等。
一般默认规则为
| section | 组成 |
| 代码段(.text) | 函数定义、程序语句 |
| 数据段(.data) | 初始化的全局变量、初始化的静态局部变量 |
| BSS段(.bss) | 未初始化的全局变量、未初始化的静态局部变量 |
使用举例:
int val0 = 8;
int val1 __attribute__((section(".data")));
int main(void)
{return 0;
}
此时我们用readelf查看,可以发现val1放在了数据段中。
二、aligned
aligned和packed是用来显式指定一个变量的存储对齐方式。aligned一般用来增大变量的地址对齐。
C语言中各种基本数据类型要按照自然边界对齐:一个char型的变量按照1字节对齐,一个short行的整型变量按照sizeof(short int)=2字节对齐,一个int型的整型变量要按sizeof(int)=4字节对齐。
C语言中不仅基本数据类型要按照自然边界对齐,复合数据类型也要按照各自的对其原则对齐。
结构体对齐原则如下:
- 结构体内各成员按照各自数据类型的对齐模数对齐。
- 结构体整体对齐方式:按照最大成员的size或其size的整数倍对齐。
联合体对齐原则如下:
- 联合体的整体大小:最大成员对齐模数或对齐模数的整数倍。
- 联合体的对齐原则:按照最大成员的对齐模数对齐。
如果你想定义一个变量,在内存中以8字节地址对齐,就可以如下:
int a __attribute__((aligned(8)));
更甚,我们可以显式指定结构体内某个成员的地址对齐,也可以显示指定整个结构体的对齐方式。如:
struct data {char a;short b __attribute__((aligned(4)));int c;
}struct data {char a;short b;int c;
}__attribute__((aligned(16)));
需要注意的是,编译器对每个基本数据类型都有默认的最大边界对齐字节数,如果超过了,编译器只能按照它规定的最大对齐字节数给变量分配地址。
总结:通过aligned属性声明,可以显示的指定变量的对齐方式,简化CPU和内存RAM之间的接口和硬件设计,但是也会因为边界对齐造成一定的内存空洞,浪费内存资源。
三、packed
aligned和packed是用来显式指定一个变量的存储对齐方式。packed一般用来减少变量的地址对齐。指定变量或类型使用最可能小的地址对齐方式。
如:
struct data {char a;short b __attribute__((packed));int c __attribute__((packed));
}struct data {char a;short b;int c;
}__attribute__((packed));
这两种方式,结构体大小都为7。对整个结构体添加packed属性和分别对每个成员添加packed属性是一样的。
在内核源码中,我们经常看到aligned和packed一起使用,这样既避免了结构体内各成员因地址对齐产生内存空洞,又指定了整个结构体的对齐方式。
struct data {char a;short b;int c;
}__attribute__((packed,aligned(8)));
这个结构体大小为8。
四、format
format属性可以指定变参函数的参数格式检查。
例如,我们实现一个自己的打印函数,为了确保传入参数的格式正确性,可以添加该属性。如:
#include <stdio.h>
#include <stdarg.h>
/*va_list:定义在编译器头文件stdarg.h中。va_start(fmt, args):根据参数args的地址,获取args后面参数的地址,并保存在fmt指针变量中va_end(args):释放args指针,将其赋值为NULL
*/void __attribute__((format(printf,1,2))) my_printf(char a, ...)
{va_list args;va_start(args, a);vprintf(a, args);va_end(args);
}int main(void)
{int num = 0;my_printf("hello world!\n", num);return 0;
}
五、weak
weak属性可以将一个强符号转换为弱符号。
使用方法如下:
void __attribute__((weak)) func(void);
int num __attribute__(weak);
强符号:函数名,初始化的全局变量名。
弱符号:未初始化的全局变量名。
使用举例:
int a __attribute__((weak)) = 1;void f(void)
{printf("f:a = %d\n", a);
}int a = 4;int main(void)
{printf("main:a = %d\n", a);f();return 0;
}
程序运行结果如下:
main:a = 4
f:a = 4
六、alias
alias属性主要用来给函数定义一个别名。
void _f(void)
{printf("_f\n");
}void f() __attribute__((alias("_f")));int main(void)
{f();return 0;
}
程序运行结果如下:
_f
通过alias属性声明,我们给_f()函数定义了一个别名f(),以后如果想要调用_f()函数,则直接通过f()调用即可。
在Linux内核中,我们会发现alias有时会和weak属性一起使用。特别是当有些函数随着内核版本升级,函数接口发生了变化,我们可以通过alias属性对这个旧的接口名字进行封装,重新起一个接口名字。
//f.c
void _f(void)
{printf("_f()\n");
}void f() __attribute__((weak, alias("_f")));//main.c
void __attribute__((weak)) f(void);
void f(void)
{printf("f()\n");
}
int main(void)
{f();return 0;
}
如果我们在main.c中重新定义了f()函数,那么当main()函数调用f()函数时,会直接调用main.c中新定义的函数;当f()函数没有被定义时,则调用_f()函数。
七、noinline和always_inline
这两个属性的用途是告诉编译器,在编译时,对我们指定的函数内联展开或不展开。
使用方法为:
static inline __attribute__((noinline)) int func();
static inline __attribute__((always_inline)) int func();
使用inline声明的函数被称为内联函数,内联函数一般会有一个static或extern修饰。使用inline声明一个内联函数,和使用关键字register声明一个寄存器变量一样,只是建议编译器在编译时内联展开。编译器会根据实际情况(函数体大小、)来做决定。使用register修饰一个变量时,只是建议编译器在为变量分配存储空间时,将这个变量放到寄存器里,使程序的运行效率更高。编译器会根据寄存器资源是否紧张,这个变量的类型及是否频繁使用来做权衡。
但是,我们使用noinline和always_inline对一个内联函数做显式属性声明时,编译器的编译行为就变得确定了:使用noinline声明,就是告诉编译器不要展开;使用always_inline属性声明,就是告诉编译器要内联展开。
相关文章:
GNU C编译器扩展关键字:__attribute__
目录 一、section 二、aligned 三、packed 四、format 五、weak 六、alias 七、noinline和always_inline GNU C增加了一个__attribute__关键字用来声明一个函数、变量或类型的特殊属性,可以知道编译器在编译过程中进行特定方面的优化或代码检查。 目前&…...
C++基础 | 从C到C++快速过渡
一、开发环境 c使用的编译器是g。 vim或者vscodeclionVS 二、C版本的Hello World /*** brief c版本helloworld示例* author Mculover666* date 2023/2/26*/#include <iostream> using namespace std;int main() {int a 1;double b 3.14;char c[] "str…...
【C++】仿函数 -- priority_queue
文章目录一、priority_queue 的介绍和使用1、priority_queue 的介绍2、priority_queue 的使用3、priority_queue 相关 OJ 题二、仿函数1、什么是仿函数2、仿函数的作用三、priority_queue 的模拟实现一、priority_queue 的介绍和使用 1、priority_queue 的介绍 priority_queu…...
盘一盘C++的类型描述符(一)
前言 C的类型描述方式是从C语言继承来的,并且进行了扩充(例如引用、非静态成员函数、模板实参等)。但由于C语言中的类型描述方式就略微有点「反人类」,再经C扩展后就有点「反碳基生物」了~ 是的,当我第一次看到这种描…...
Peppol的发展史和基本框架
Peppol(Pan-European Public Procurement Online)是欧洲区域内的一个跨境公共采购电子商务平台试点项目,由欧盟委员会和Peppol联盟成员国共同资助建立,旨在通过制定标准化框架,推动欧盟成员国在公共采购相关的电子目录…...
Linux-GCC介绍+入门级Makefile使用
前言(1)我们都知道,在Linux中编译.c文件需要使用gcc -o .c文件的指令来将C文件变成可执行文件。但是我们有没有发现,如果我们需要编译大一点的工程,后面需要加上的.c文件是不是太多了?感觉非常的麻烦。&…...
iOS(一):Swift纯代码模式iOS开发入门教程
Swift纯代码模式iOS开发入门教程项目初始化(修改为纯代码项目)安装第三方库(以SnapKit库为例)桥接OC库(QMUIKit)封装视图并进行导航跳转示例:使用 TangramKit 第三方UI布局库应用国际化添加 R.s…...
IDEA+Python+Selenium+360浏览器自动化测试
环境配置前提,见文章https://mp.csdn.net/mp_blog/creation/editor/new?spm1001.2101.3001.4503下载360浏览器,并下载对应版本的chromedriver.exe,下载地址http://chromedriver.storage.googleapis.com/index.html下载好360浏览器࿰…...
运输层概述及web请求
运输层 运输层概述 运输层向高层用户屏蔽了下面网络核心的细节(如网络拓扑、所采用的路由选择协议等)它使应用进程看见的就好像是在两个运输层实体之间有一条端到端的逻辑通信信道; 根据需求不同,运输层提供两种运输协议 面向连…...
python与pycharm从零安装
python(解释器)下载地址:Welcome to Python.orgpycharm(编译器)下载地址:PyCharm: the Python IDE for Professional Developers by JetBrains一、python的下载与安装到官网后根据步骤下载安装包后…...
叠氮试剂943858-70-6,Azidobutyric acid NHS ester,叠氮-C3-活性酯
1、试剂基团反应特点(Reagent group reaction characteristics):Azidobutyric acid NHS ester具有叠氮化物和NHS酯端基。西安凯新生物科技有限公司供应的叠氮化物可以与炔烃、DBCO和BCN进行铜催化的点击化学反应。NHS酯可以与胺基反应&#x…...
pycharm激活虚拟环境时报错:无法加载文件activate.ps1,因为在此系统上禁止运行脚本,Windows10系统
问题: ii_env\Scripts\activate : 无法加载文件 F:\gitlab\AutoFrame\ii_env\Scripts\Activate.ps1,因为在此系统上禁止运行脚本。 有关详细信息,请参阅 https:/go.microsoft.com/fwlink/?LinkID135170 中的 about_Execution_Policies。 所在…...
刷题小抄4-数组
在Python中数组的功能由列表来实现,本文主要介绍一些力扣上关于数组的题目解法 寻找数组中重复的数字 题目链接 题目大意: 给出一个数组,数组长度为n,数组里的数字在[0,n-1]范围以内,数字可以重复,寻找出数组中任意一个重复的数字,返回结果 解法一 该题最基础的思路是使用字…...
Hbase安装
目录 上传压缩包 解压 改名 修改 Hbase 配置文件 修改base-env.sh 修改hbase-site.xml 配置环境变量 修改zookeeper配置文件 复制配置文件 修改zoo.cfg配置文件 修改myid 配置环境变量 刷新配置文件 启动Hbase 进入Hbase 查看版本号 查看命名空间 查看命名空…...
面向对象设计模式:结构型模式之代理模式
一、引入 访问 FB:代理服务器 二、代理模式 aka Surrogate 2.1 Intent 意图 Provide a surrogate (代理) or placeholder for another object to control access to it. 为另一个对象提供一个代理或占位符,以控制对它的访问。代理模式给某一个对象提…...
CCF大数据专家委员会十周年纪念庆典纪实:拥抱数字时代,展望科技未来
山河远阔,奋进十年,作为国内大数据领域最权威的学术组织,CCF大数据专家委员会(以下简称“大专委”)不忘初心,凝心聚力,见证并推动了过去10年来大数据技术生态在中国的建立、发展和成熟。 2023年…...
Qt学习3-Qt Creator四则运算计算器(哔站视频学习记录)
计算器中的“”按钮这部分的代码解释 目录 制作计算器中的“”按钮这部分的代码解释 一、代码部分 二、解释 三、思路 四、死循环! 一、代码部分 void Widget::on_equalButton_clicked() {QStack<int> s_num,s_opt; //声明两个int类型变量char opt[128…...
学习 Python 之 Pygame 开发魂斗罗(九)
学习 Python 之 Pygame 开发魂斗罗(九)继续编写魂斗罗1. 在子弹类中修改敌人发射子弹的位置2. 创建显示敌人子弹的函数3. 解决敌人不会向下掉落的问题4. 给敌人碰撞体组增加碰撞体5. 解决敌人叠加在一起的问题继续编写魂斗罗 在上次的博客学习 Python 之…...
最简单的SpringBoot+MyBatis多数据源实现
最简单的SpringBootMyBatis多数据源实现1.数据库准备2.环境准备3.代码部分3.1多数据源配置2.测试随着应用用户数量的增加,相应的并发请求的数量也会跟着不断增加,慢慢地,单个数据库已经没有办法满足频繁的数据库操作请求了,在某些…...
Spring Boot 3.0系列【8】核心特性篇之SpringApplication
有道无术,术尚可求,有术无道,止于术。 本系列Spring Boot版本3.0.3 源码地址:https://gitee.com/pearl-organization/study-spring-boot3 文章目录 前言1. 启动应用2. 自定义 Banner3. 应用参数传递参数获取参数4. ApplicationRunner、CommandLineRunner5. 事件发布和监听…...
使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式
一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明:假设每台服务器已…...
visual studio 2022更改主题为深色
visual studio 2022更改主题为深色 点击visual studio 上方的 工具-> 选项 在选项窗口中,选择 环境 -> 常规 ,将其中的颜色主题改成深色 点击确定,更改完成...
Objective-C常用命名规范总结
【OC】常用命名规范总结 文章目录 【OC】常用命名规范总结1.类名(Class Name)2.协议名(Protocol Name)3.方法名(Method Name)4.属性名(Property Name)5.局部变量/实例变量(Local / Instance Variables&…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)
引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...
WEB3全栈开发——面试专业技能点P2智能合约开发(Solidity)
一、Solidity合约开发 下面是 Solidity 合约开发 的概念、代码示例及讲解,适合用作学习或写简历项目背景说明。 🧠 一、概念简介:Solidity 合约开发 Solidity 是一种专门为 以太坊(Ethereum)平台编写智能合约的高级编…...
鱼香ros docker配置镜像报错:https://registry-1.docker.io/v2/
使用鱼香ros一件安装docker时的https://registry-1.docker.io/v2/问题 一键安装指令 wget http://fishros.com/install -O fishros && . fishros出现问题:docker pull 失败 网络不同,需要使用镜像源 按照如下步骤操作 sudo vi /etc/docker/dae…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
html css js网页制作成品——HTML+CSS榴莲商城网页设计(4页)附源码
目录 一、👨🎓网站题目 二、✍️网站描述 三、📚网站介绍 四、🌐网站效果 五、🪓 代码实现 🧱HTML 六、🥇 如何让学习不再盲目 七、🎁更多干货 一、👨…...
【VLNs篇】07:NavRL—在动态环境中学习安全飞行
项目内容论文标题NavRL: 在动态环境中学习安全飞行 (NavRL: Learning Safe Flight in Dynamic Environments)核心问题解决无人机在包含静态和动态障碍物的复杂环境中进行安全、高效自主导航的挑战,克服传统方法和现有强化学习方法的局限性。核心算法基于近端策略优化…...
