C/C++杂谈-printf的可变参数机制
C/C++杂谈-printf的可变参数机制
文章目录
- C/C++杂谈-printf的可变参数机制
- printf的使用
- printf的源码
- 源码剖析
- 多参数实现机制原理
C++11引入了可变参数模板机制,对模板参数进行了高度泛化,但是对于可变参数其实C语言学习中早已遇到过,那就是printf可以进行多参数的输出,这是怎么实现的呢?
printf的使用
我们对于printf的用法无非两种
const char *str = "hello , world\n";printf(str);//直接传入字符串地址int year = 2023;printf("%d%s", year, "原神启动");//传入格式控制字符串地址和参数
我们printf的参数是先是一个字符串,后面才是我们的输出变量,可以嗅出printf对于多参数的控制应该和传入的第一个字符串有关,那么究竟是如何实现的呢?
printf的源码
//acenv.h
typedef char *va_list;
#define _AUPBND (sizeof (acpi_native_int) - 1)
#define _ADNBND (sizeof (acpi_native_int) - 1)#define _bnd(X, bnd) (((sizeof (X)) + (bnd)) & (~(bnd)))
#define va_arg(ap, T) (*(T *)(((ap) += (_bnd (T, _AUPBND))) - (_bnd (T,_ADNBND))))
#define va_end(ap) (void) 0
#define va_start(ap, A) (void) ((ap) = (((char *) &(A)) + (_bnd (A,_AUPBND))))
//start.c
static char sprint_buf[1024];
int printf(char *fmt, ...)//格式控制字符串和C的函数多参数
{va_list args;//va_list就是char * 的typedefint n;va_start(args, fmt);n = vsprintf(sprint_buf, fmt, args);va_end(args);write(stdout, sprint_buf, n);return n;
}
//unistd.h
static inline long write(int fd, const char *buf, off_t count)
{return sys_write(fd, buf, count);
}
源码剖析
映入眼帘的就是一串宏定义
看我们printf的函数参数部分,char *fmt就是我们的格式控制字符串,后面的…是C的函数多参数,即后面的参数数目不定
va_list就是char * 的typedef,也就是定义了名为args的char指针
va_start(args, fmt);就是把args指向fmt后面的第一个参数的地址
这里对va_start进行解释
#define va_start(ap, A) (void) ((ap) = (((char *) &(A)) + (_bnd (A,_AUPBND))))
va_start(ap, A) (void) ((ap) = (((char *) &(A)) + (_bnd (A,_AUPBND))))
可见ap指向的是A地址往后偏移_bnd字节,而 _bnd 传参了 _ADNBND,_ADNBND = (sizeof (acpi_native_int) - 1)
typedef s32 acpi_native_int 若采用这种宏定义,表明int 类型是用32位表示,也表示当前内存是4字节对齐
acpi_native_int这个参数是平台相关的
所以sizeof (acpi_native_int)是当前平台的int大小,我们假设是4字节,那么_ADNBND就是3
#define _bnd(char,3) ==> (1+3)&(~3) ==> 4#define _bnd(int,3) ==> (4+3)&(~3) ==> 4#define _bnd(double,3) ==> (4+3)&(~3) ==> 8
我们通过上述样例可以明白**_bnd就是获取类型A的内存对齐大小**,假如32位平台那么就是4的倍数
所以va_start(args, fmt) 就是把fmt偏移char*内存对齐大小个字节然后赋值给args,这样args指向的就是格式字符串后面的参数
n = vsprintf(sprint_buf, fmt, args);这里的n则是我们实际控制输出的字符数,我们printf实际就是一个输出字符的函数,n也是我们的返回值
而后面的 write(stdout, sprint_buf, n);无非就是把缓冲区里的n个字符输出到stdout输出流,这就不是我们讨论的重点了
多参数实现机制原理
通过上面的剖析,我们发现printf由格式控制字符串得到下一个参数的起始地址,而下一个起始地址是fmt地址偏移内存对齐大小个字节
这是为什么呢?
这跟函数的压栈顺序有关。我们C/C++默认__cdel的从右至左将参数压栈,而我们栈是向下增长的,所以先入栈的地址高,后入栈的地址低,所以格式字符串的地址最低,往上偏移自然能得到其他参数的地址
void func(int a, int b, int c)
{printf("a = %d located [%x]\n", a, &a);printf("b = %d located [%x]\n", b, &b);printf("c = %d located [%x]\n", c, &c);
}
signed main()
{func(1, 2, 3);return 0;
}
//输出
a = 1 located [b3bff960]
b = 2 located [b3bff968]
c = 3 located [b3bff970]
得到地址后,由于我们规定格式控制字符串中%的数量即为输出参数数量,然后就能拿到所有参数放到缓冲区,再输出到标准输出流
如果我们想要实现多参数机制(需要了解<stdarg.h>),自然也要通过我们的参数设定模式,类似格式控制中百分号的数量来确定参数的数目,而名称出现的顺序对应参数的顺序。
可见C语言的多参数机制是很繁琐的,而我们C++11引入可变参数模板也正是为了追求更好的参数泛化。
相关文章:
C/C++杂谈-printf的可变参数机制
C/C杂谈-printf的可变参数机制 文章目录 C/C杂谈-printf的可变参数机制printf的使用printf的源码源码剖析 多参数实现机制原理 C11引入了可变参数模板机制,对模板参数进行了高度泛化,但是对于可变参数其实C语言学习中早已遇到过,那就是printf…...
es基本语法 (kibana)
#添加 (不添加id默认会生成id) POST /cj/test {"name":"jack","sex":"1","age":12 } #添加 (id为第5个的) POST /cj/test/5 {"name":"jackson","sex":"1","age":12 } #条…...
Tesco EDI需求分析
Tesco,成立于1919年,是一家全球领先的综合性零售企业,总部位于英国。公司致力于提供高质量、多样化的商品和服务,以满足客户的需求。Tesco的使命是通过创新和卓越的客户服务,为客户创造更美好的生活。多年来࿰…...
html常用的标签
基本结构标签 <!DOCTYPE>: 定义 HTML 文档类型。<html>: HTML 文档的根元素。<head>: 文档的头部,包含了元数据和引用的外部资源。<title>: 定义网页标题,显示在浏览器标签上。&l…...
护眼灯什么价位的好?适合学生入手的护眼台灯推荐
据60年前的统计,中国人口的近视率约为10%至20%。 国家卫健委发布的中国首份眼健康白皮书显示,我国小学生近视率为47.2%,初中生近视率为75.8%,大学生近视率超过90%。如今,“低头族”随处可见,近视人群日益增…...
大数据架构
大数据架构 https://huaweicloud.csdn.net/633578fed3efff3090b58398.html https://blog.csdn.net/yuanziok/article/details/117030031 https://blog.csdn.net/qq_46675545/article/details/121985987 https://blog.csdn.net/qq_33367934/article/details/127685417 https://b…...
【Linux】C文件系统详解(四)——磁盘的物理和抽象结构
文章目录 磁盘结构磁盘物理结构磁盘的具体物理结构磁盘结构的逻辑抽象 文件系统BootBlockSuperBlockGroupDescriptorTableinode tableDataBlocksinodeBitmapblockBitmaplinux中的inode 和文件名如何理解文件的增删查改删 补充细节1.如果文件误删了,我们该怎么办?2.inode确定分…...
论文-分布式-拜占庭将军问题
目录 0-前言 1-导引 2-不可能性 3将军(1叛徒)问题不存在解/不能达成共识 少于3m1个将军(有m个叛徒)不存在解/不能达成共识 精确一致性与近似一致性是同等困难的 3-使用口头消息的解 “口头消息”的含义 OM(m)算法的步骤 OM(m)算法的正确性推导 4-使用签名消息情况下…...
万字解析设计模式之 适配器模式
一、 适配器模式 1.1概述 将一个接口转换成客户希望的另一个接口,适配器模式使接口不兼容的那些类可以一起工作。 适配器模式分为类适配器模式和对象适配器模式,前者类之间的耦合度比后者高,且要求程序员了解现有组件库中的相关组件的内部结…...
Linux 安全 - 扩展属性xattr
文章目录 前言一、简介二、扩展属性命名空间2.1 简介2.2 security扩展属性2.3 System扩展属性2.4 Trusted扩展属性2.5 User扩展属性 三、用户空间使用3.1 setfattr/getfattr3.2 setxattr/getxattr/listxattr 参考资料 前言 一、简介 xattr - Extended attributes扩展属性是与…...
spring boot加mybatis puls实现,在新增/修改时,对某些字段进行处理,使用的@TableField()或者AOP @Before
1.先说场景,在对mysql数据库表数据插入或者更新时都得记录时间和用户id 传统实现有点繁琐,这里还可以封装一下公共方法。 2.解决方法: 2.1:使用aop切面编程(记录一下,有时间再攻克)。 2.1.1&am…...
我的创作纪念日2048天
机缘 在这特殊的日子里,我要庆祝我的 CSDN 创作纪念日——已经坚持了整整2048天! 在这2048天里,我经历了很多成长和收获。作为一名技术写手,我投入了大量的时间和精力来分享我的知识和经验。我曾经写过关于数据库、数据同步、数…...
MatrixOne实战系列回顾 | 导入导出项目场景实践
本次分享主要介绍MatrixOne导入导出以及项目场景实践。将从四个方向为大家演示MatrixOne的功能,分别是数据的导入、导出、对接数据集成工具,以及Java连接实战。 数据导入会使用三种方式将数据导入至 MatrixOne中。分别是insert语句、load data语句还有s…...
Find My音箱|苹果Find My技术与音箱结合,智能防丢,全球定位
音箱市场规模正在不断扩大。随着人们生活品质的提高,对音乐体验的需求也在不断升级。消费者对于蓝牙音箱的需求,已经从单纯的音质扩展到了功能、设计和价格等多个方面。随着移动化、即时化的视听娱乐需求的增长,蓝牙音箱性能、质量、外观设计…...
51单片机应用
目录 编辑 1. C51的数据类型 1.1 C51中的基本数据类型 1.2 特殊功能寄存器类型 2. C51的变量 2.1 存储种类 1. C51的数据类型 C51是一种基于8051架构的单片机,它支持以下基本数据类型: 位(Bit):可以表…...
系列三、ThreadLocal vs synchronized
一、ThreadLocal vs synchronized 虽然ThreadLocal与synchronized关键字都能用于处理多线程并发访问变量的问题,但是两者处理问题的角度和思路是不一样的。区别如下: 小总结:虽然上一篇中的案例都实现了线程隔离,但是使用ThreadLo…...
封装Redis工具类
基于StringRedisTemplate封装一个缓存工具类,满足下列需求: 方法1:将任意Java对象序列化为json并存储在string类型的key中,并且可以设置TTL过期时间 方法2:将任意Java对象序列化为json并存储在string类型的key中&…...
使用 millis() 函数作为延迟的替代方法(电位器控制延迟时间)
接线图: 代码: unsigned long currentMillis 0; unsigned long previousMillis_LED1 0; unsigned long LED1_delay0; unsigned long previousMillis_LED2 0; unsigned long LED2_delay0; #define LED1 3 #define LED2 9 #define P1 A2 …...
MySQL之BETWEEN AND包含范围查询总结
一、时间范围 查询参数格式与数据库类型相对应时,between and包含头尾,否则依情况 当数据库字段中存储的是yyyy-MM-dd格式,即date类型: 用between and查询, 参数yyyy-MM-dd格式时,包含头尾,相当…...
LangGraph生态全景:Python Agent开发指南
先给你一个“全景图”,把 LangGraph 生态的主要组件拆开讲一下(只列和你做 Python Agent 开发最相关的)。 #mermaid-svg-Rqe3jXYezkcPijBL{font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}@keyfr…...
D2DX终极指南:让《暗黑破坏神2》在现代PC上焕然新生的完整教程
D2DX终极指南:让《暗黑破坏神2》在现代PC上焕然新生的完整教程 【免费下载链接】d2dx D2DX is a complete solution to make Diablo II run well on modern PCs, with high fps and better resolutions. 项目地址: https://gitcode.com/gh_mirrors/d2/d2dx D…...
7个技巧彻底搞懂esbuild中switch语句的解析机制
7个技巧彻底搞懂esbuild中switch语句的解析机制 【免费下载链接】esbuild An extremely fast bundler for the web 项目地址: https://gitcode.com/GitHub_Trending/es/esbuild esbuild作为一款超快速的Web打包工具,其高效的JavaScript解析能力是实现极速构建…...
Java 数组基础知识
一、数组定义及基础知识1、数组是同类型数据的有序集合一次性存多个相同类型的数据长度固定不可变每个元素有下标(索引),从 0 开始2、语法格式:int[] array;double[] array;boolean[] array;String[] array;Object[] array;//数组…...
新粗野主义React组件库:从设计原理到工程实践
1. 项目概述:当“新粗野主义”撞上组件库 如果你是一个前端开发者,或者对现代网页设计趋势有所关注,最近可能被一种名为“新粗野主义”的设计风格刷屏。它大胆、直接、甚至有些“粗糙”,用高饱和度的色彩、粗重的边框、不加修饰的…...
倒计时90天!SITS2026新规强制要求AISMM评估质量追溯机制,3类组织正紧急补签质量承诺书
更多请点击: https://intelliparadigm.com 第一章:SITS2026专家:AISMM评估质量保障 AISMM(AI Software Maturity Model)是SITS2026国际标准中用于衡量AI系统工程化成熟度的核心框架,其评估质量直接决定组织…...
AISMM基准数据首次全球统一发布(SITS2026核心机密解封)
更多请点击: https://intelliparadigm.com 第一章:SITS2026发布:AISMM行业基准数据 SITS2026 是面向智能交通系统(ITS)与多模态感知融合领域发布的全新行业基准数据集,由 AISMM(Autonomous In…...
互联网大厂 Java 求职面试:从 Java SE 到 Spring Boot 的技术探讨
互联网大厂 Java 求职面试:从基础到复杂的技术考察 在这个故事中,我们将跟随两位角色:面试官与燕双非,一位搞笑的程序员。他们将在互联网大厂的面试现场进行一场精彩的对话。第一轮提问 面试官(严肃)&#…...
D-Compress:面向机器感知的LiDAR点云实时压缩技术
1. 项目概述在资源受限的机器人系统中,实时传输和处理LiDAR点云数据一直是个棘手的问题。想象一下,一个自主导航的机器人需要将周围环境的3D点云数据实时传输到边缘服务器进行处理,但受限于有限的网络带宽和计算资源,原始点云数据…...
前端组件开发公众号产品推广与生态共建方案
摘要: 在数字化浪潮席卷全球的背景下,前端技术作为连接用户与数字服务的关键桥梁,其发展速度与应用广度持续拓展。与此同时,围绕前端技术构建的开发者社区正逐步演变为技术传播、产品孵化与商业合作的重要载体。本文以“前端组件开…...
