C语言可变参数只会用算啥本事?看我来抽丝剥茧干翻它!

看山是山,看山不是山,最终看山才是山,并且是无穷的山峦。当我们学习一门技术的时候,起初是先模仿,但是最终是为了超越,也就是得到秘籍,看到本质。

于是,今天来继续看可变参数,我们来分析这个过程,代码如下:

声明int add (int num,...);...会说明是个可变函数,这样子编译器在编译遇到的函数的地方,就知道自动解析,依据传入的参数,直接进行进栈,从而不需要报错。

我们默认的函数,如果声明是两个参数,调用的是三个参数的话,最终会出错,会提示你找不到实现体。

声明完了后,实现函数,然后我们在使用的地方,直接传入多个参数,就不会出问题。

我们这里要看下add里面的具体实现,第一个参数我们利用这个值,用来判定长度,循环的结束点。

然后使用va_start 来卡找到第一个参数的地址,我这节画个图,大家就能理解了。再一个你也就明白了,为什么是int add(int num,...);而不是int add(...); 因为如果是这个,你实现的时候,就没法定位到起始地址,导致你无法解析后续的参数,这个明白了吧。

所以,编译时候,系统会提示错误,规定...参数前面必须有一个有名参数,否则系统编译的时候,你在实现体里面,无法做处理。这里就是 va_start(valist,num);

然后我们使用va_arg 去获取后续的参数,第一个是起始地址,第二个是后面紧跟着的数据,该以哪个大小去解析。我们这里是int,都是这个尺寸,所以用了循环。

在printf里面,使用的是%s ,%f这类处理,依据这个会变化。

使用完成后,就可以用va_end来结束指针。

我们如果把调用地方改成add(5,a,b);最终能输出出来结果,但是非常乱,原因很简单,这里的第一个参数5,让遍历寻找了栈上面的一些脏数据,导致结果未知。

我们如果把int b=6改成float b=6;会发生什么神奇的现象呢?出现了神奇的结果,原因很简单,float 进入栈的时候,占用的空间比int大,但是我们执行的时候,用了int大小去解析了这个数据,导致出现问题。

我们来看下如何修正这个问题,就需要格式化处理了,第一个参数,我们把它调整下,变成char *format,我们把代码改成这个,

为什么这里有float变为了double,主要是方便系统进行处理,升级后你 解析的时候,就要用double去解析,否则的话你处理完数据,ap指针就没有指到下一个位置,导致出错。

不过这个问题现在你不需要担心,如果你没有写对,系统会在编译时候提醒你,直接系统报错,让你去修改的。

这样子看下来,是不是觉得可变参数也没多神奇了?简单说下就是编译器支持...让函数参数可变,同时保留一个有名参数,让实现体可以用这个去定位到起始位置,然后进行遍历解析,完成逻辑处理。

我们把这个再抽象一层,简单来说,就是一组数据约束,存放在一起,然后我们依据一个格式化参数,对这个数据进行解析处理。当你看到这个的时候,就突然明白,协议的概念。

协议就是约定,约定双方同时遵循一个规则,同时遵守,就是协议。TCP/IP协议,ELF文件解析协议。

当你抽象到这里,基本上就大彻大悟,一切处理都是协议头 数据( 校验)。

在今天最后,我们来看下反编译后的代码,就明白了第一个参数的地址的意义,我这里用的int b=6;原因是如果是float的话,指令会比较复杂,不方便我们学习。

这里可以看到,我们地参数是 %d,%d  a b  这三个,在汇编代码中,可以看到,

rbp-0x8 放的是b的值,6

rbp-0xc 放的是a的值,5

rbp-0x14,放着一个地址4007ec,这地方就是%d,%d 常量字串位置

所以我们add函数实现里面,fmt拿到的就是rbp-0x14的地址,也就是第一个参数,随后的解析就是依据给的格式,把对应数据解析出来,移动指针ap的位置到合适地方(依据对齐原则,以及sizeof(数据类型)

声明:

本文于网络整理,版权归原作者所有,如来源信息有误或侵犯权益,请联系我们删除或授权事宜。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/338670.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java8升级java12_为什么现在是升级到Java 8的最佳时机

java8升级java12有兴趣了解如何通过AppDynamics充分利用Java 8的新功能吗? 立即开始免费试用 ! 今年3月,Oracle发布了近十年来最受期待的版本Java8。自发布以来,最新版本引起了越来越多的关注,各种规模的公司都渴望升…

C语言#include还有些你不知道的事

#include简介在C语言中#include是preprocessor的一条指令,告诉预处理器将指定头文件的内容插入到预处理器命令的相应位置。#include "xxx.h" 和 #include有两种方式可以指定插入头文件:#include #include "filename"如果需要包含标…

java常见的ide_在三个Java IDE中生成的三种常见方法

java常见的ide在本文中,我研究了NetBeans 8.0.2 , IntelliJ IDEA 14.0.2和Eclipse Luna 4.4.1生成的三种“通用”方法[ equals(Object) , hashCode()和toString() ]的区别…

深度linux安装依赖,Linux -- Ubuntu下载deepin wine依赖问题笔记

问题开始下载deepin-wine安装包, 请稍后…1.1udis86_1.72-2_i3 100%[>] 34.18K 87.3KB/s 用时 0.4s1.2deepin-fonts-win 15%[> ] 31.18K 1.72KB/s 用时 18s1.2deepin-fonts-win 100%[>] 207.88K 26.2KB/s 用时 6.7s2.1deepin-libwine_2 100%[>] 18.97M 132KB/s 用时…

什么是C语言中的隐式函数声明?

「1、什么是C语言的隐式函数声明」在C语言中,函数在调用前不一定非要声明。如果没有声明,那么编译器会自动按照一种隐式声明的规则,为调用函数的C代码产生汇编代码。下面是一个例子:int main(int argc, char** argv) {double x a…

群晖 上传 源文件不存在_群晖NAS连接百度网盘报错?原因是这样的

群晖NAS附带的云同步套件可以与国内外多个网盘连接 , 连接后可从云上下载数据亦可从本地将数据上传到云上。例如通过云同步套件连接百度网盘账号后可以便捷上传和下载数据 , 若网盘空间较大甚至可用来备份整个NAS等。不过现在看来群晖与百度网盘的合作似乎已经结束,…

ssl/tls服务器瞬时_SSL / TLS REST服务器–带有Spring和TomEE的客户端

ssl/tls服务器瞬时在构建系统时,开发人员通常会忽略安全性方面。 安全一直是令人担忧的重要问题,但是它比以前吸引了更高的关注。 就在今年,我们发生了像Heartbleed Bug或CelebrityGate丑闻这样的案件。 这与帖子无关,只是安全真正…

linux kvm百度云,容器与云|如何在 Ubuntu Linux 上使用 KVM 云镜像

如何下载并使用运行在 Ubuntu Linux 服务器上的 KVM 云镜像?如何在 Ubuntu Linux 16.04 LTS 服务器上无需完整安装即可创建虚拟机?如何在 Ubuntu Linux 上使用 KVM 云镜像?基于内核的虚拟机(KVM)是 Linux 内核的虚拟化模块,可将其…

C 的16个大坑,你能躲过几个?

首先说下C 和C语言有什么区别?分享一个我在知乎上看见的回答:C ≈ C with classes, C with STLC:面向机器编程C :面向编译器编程C 有个很重要的特性叫RAII,个人认为可以多多使用,相当方便。言归…

java 性能调优_Java性能调优调查结果(第三部分)

java 性能调优这是本系列文章的第三篇,我们将分析2014年10月进行的调查的结果。如果您尚未这样做,我建议从本系列的前两篇文章开始: 问题严重性分析和监视域分析 。 这篇文章着重于故障排除/根本原因检测。 本调查部分的背景:意识…

不懂指针类型,7个例子给你讲明白

1. int va;这是一个整型变量,32位CPU的话,占有32个bite2. int *va;这是一个整型指针变量,用于存放一个整型变量的地址,3. int **va;这是一个整型的二级指针,用于存放一个内存的地址,该地址对应的内存中存放…

Tomcat与Netty比较

Tomcat介绍Tomcat支持的协议Tomcat的优缺点Netty介绍Netty支持的协议Netty的优点和缺点Tomcat和Netty的区别Tomcat和Netty的应用场Tomcat和Netty来处理大规模并发连接的优化Tomcat与Netty的网络模型的区别Tomcat与Netty架构设计拓展 Tomcat介绍 Tomcat是一个免费的、开放源代码…

C或C 如何通过程序执行shell命令并获取命令执行结果?

1 应用场景最近在实际程序开发中,需要通过程序执行 shell 命令,并获取命令输出内容。但是系统自带的 system 只能返回命令执行成功与否,不能捕获命令输出。2 扩展性由于应用场景本就广泛,因此扩展性较好。此函数可以执行任意命令&…

linux centos7安装ngix,centos7 环境下安装nginx--Linux

本文将要为您介绍的是centos7 环境下安装nginx--Linux,具体完成步骤:一、安装前需要的编译环境准备1、安装makeyum install -y gcc automake autoconf libtool make2、安装gcc、gcc-cyum install -y gcc gcc-c3、关闭防火墙iptables -F4、关闭selinux#临时关闭:sete…

primefaces_使用PrimeFaces开发数据导出实用程序

primefaces我的日常工作涉及大量使用数据。 我们使用关系数据库来存储所有内容,因为我们依赖于企业级的数据管理。 有时,具有将数据提取为简单格式(例如电子表格)的功能很有用,以便我们可以按需进行操作。 这篇文章概述…

如何优雅地实现判断一个值是否在一个集合中?

如何判断某变量是否在某个集合中&#xff1f;注意&#xff0c;这里的集合可能并不是指确定的常量&#xff0c;也可能是变量。版本0#include int main(){int a 5;if(a 1 || a 2 || a 3 || a 4 || a 5){std::cout<<"find it"<<std::endl;}return 0;…

骚操作:利用强弱符号制作插件库

当有强符号和弱符号时&#xff0c;选择使用强符号那么我们正可以利用这个原则做以下事情&#xff1a;定义为弱符号&#xff0c;如果是弱符号&#xff0c;使用默认行为如果链接了库&#xff0c;是强符号&#xff0c;则使用外部定义行为以此来实现一个类似插件的功能。通俗一点说…

c语言里 t是什么作用,c语言里的\t是什么意思

c语言里的&#xff3c;t是什么意思以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容&#xff0c;让我们赶快一起来看一下吧&#xff01;c语言里的&#xff3c;t是什么意思正宗叫法是“水平制表符”&#xff0c;就是在输出媒体上水平跳过多个空格…

函数或全局变量重复定义时会怎样?

可能有些朋友第一反应是&#xff0c;那肯定是编译不过喽&#xff1a;// fun.c #include void func() {printf("编程珠玑\n"); }// main.c #include void func() {printf("公众号\n"); } int main(void) {func();return 0; }编译&#xff1a;$ gcc -o main …

当C语言函数执行成功时,返回1和返回0究竟哪个好?

基本上&#xff0c;没有人会将大段的C语言代码全部塞入 main() 函数。更好的做法是按照复用率高&#xff0c;耦合性低的原则&#xff0c;尽可能的将代码拆分不同的功能模块&#xff0c;并封装成函数。C语言代码的组合千变万化&#xff0c;因此函数的功能可能会比较复杂&#xf…