可变参数(c/c++)

        

目录

一、C语言版本

 二、C++的实现方法

2.1数据包

2.2sizeof...运算符

2.3可变参数模板的使用

2.4emplace_back()


        有时候我们在编写函数时,可能不知道要传入的参数个数,类型 。比如我们要实现一个叠加函数,再比如c语言中的printf,c++中的emplace_last()。

那么这些函数是如何实现的呢?

一、C语言版本

在 C 中,可变参数通过 <stdarg.h> 头文件中的宏来处理。最常用的宏是 va_list、va_start、va_arg 和 va_end。以下是这些宏的简要说明:

va_list:用于声明一个可变参数列表的类型。

其实va_list就是一个char*类型,但具体实现取决于编译器和平台。它的内部结构是由编译器实现的,对于程序员来说是不透明的。

 va_start:用于初始化一个可变参数列表,将其与函数参数列表中的最后一个固定参数关联。(因为形参是从右往左入参的,也就是右边的参数是高地址,左边的函数是低地址)

va_arg:用于从可变参数列表中读取一个参数,并指定其类型。

va_end:用于清理可变参数列表,结束可变参数的使用 

 下面我们将结合一段代码来简单的讲解

#include<iostream>
#include<stdarg.h>
int  addsum(int num, ...)
{va_list args;va_start(args, num);int ret=0;for (int i = 0; i < num; i++){int temp = va_arg(args, int);ret+=temp;}va_end(args);return ret;
}int main()
{std::cout<<addsum(5, 1, 2, 3, 4, 5);return 0;
}

在 C 语言中,如果你使用了 va_start 宏来初始化可变参数列表,那么你至少需要传递一个参数作为固定参数,以便确定可变参数列表的起始位置。这个固定参数通常被称为 "sentinel" 或 "sentinel value"。

那么这个"sentinel" 或 "sentinel value"。一定要是参数个数吗?

当然不是,从printf中我们就知道第一个参数也可以是字符串。

在实现可变参数函数时,并不一定需要传递一个表示参数个数的额外参数。额外的参数可以帮助函数确定参数的数量,但并不是必须的。实际上,很多情况下都可以通过其他方式来确定参数的数量。

以下是一些确定参数数量的方法:

  1. 约定特定的参数结尾标志:例如,C 标准库中的 printf 函数就是通过字符串中的格式化标志(例如 %d%s 等)来确定参数的数量的。

  2. 利用特定的参数类型:例如,如果所有的参数都是相同类型的,你可以在函数中使用特定的参数类型来确定参数的数量。

  3. 使用额外参数传递参数数量:虽然不是必须的,但在某些情况下,通过额外的参数传递参数的数量是一种方便的做法。

在实际应用中,选择哪种方法取决于函数的使用场景和需求。如果函数的参数数量不固定,并且无法通过其他方式确定参数的数量,那么传递一个表示参数数量的额外参数是一种常见的做法。但在某些情况下,其他方法可能更加合适。

总的来说,并不是一定要传递表示参数个数的额外参数,具体是否需要取决于函数的设计和实现需求。

而va_start其实就是将自己定义的va_list 类型的参数向后移动一个位置

在上面的代码中其实就是让args指向如图所示位置。

而va_arg就是将后面的参数从其相应的类型提取出来。这下,你就知道为什么printf中为什么要有传入%d%f这些东西了吧。(当然这些东西也有确定参数个数的作用)。

最后只剩下va_end,用于标记可变参数列表的结束。它的存在是为了确保在使用完可变参数列表后正确释放资源,以避免内存泄漏和其他潜在的问题。

在可变参数函数中,通常会使用 va_start 来初始化 va_list 对象,然后使用 va_arg 来逐个读取参数,直到参数列表的末尾。一旦处理完所有参数,就应该调用 va_end 来清理 va_list 对象,以释放相关资源。

va_end 的作用包括:

  1. 清理资源va_list 对象可能会占用一些资源,例如在某些实现中可能分配了内存。调用 va_end 可以释放这些资源,避免内存泄漏。

  2. 标记列表的结束:调用 va_end 可以显式地标记可变参数列表的结束,使得程序能够正确地识别参数列表的边界,避免访问超出列表范围的参数。

  3. 与平台相关的清理工作va_end 可能会执行与平台相关的清理工作,以确保系统资源得到正确的释放。

在使用可变参数函数时,特别是在处理可变参数列表的末尾时,始终记得调用 va_end 是很重要的。不调用 va_end 可能会导致资源泄漏和未定义的行为,因此要确保在使用完可变参数列表后及时调用 va_end

 二、C++的实现方法

2.1数据包

在 C++ 中,也可以使用可变参数模板来实现类似的功能,这种技术更加灵活,并且不需要使用宏。C++11 引入了新的语法和标准库支持,使得可变参数模板更加易用和安全。

c++在c++11中提出了可变参数模板的概念,所谓可变参数模板就是一个接受可变数目参数模板的函数或模板类。可变数目的参数被称作参数包。存在两种参数包:

1.模板参数包:表示0或多个模板参数

2.函数参数包:表示0或多个函数参数

我们使用“...”来表示一个包,在一个模板参数列表中,class..或typname...表示接下来 的参数表示零个或多个类型的列表;一个类型名后面跟一个省略号表示零个或多个给定类型的非类型参数的列表。在函数参数列表中,如果一个参数的类型是一个模板参数包,则此参数也是一个函数参数包。例如:

//Args是一个模板参数包;rest是一个函数参数包
//Args表示零个或多个模板类型参数//rest表示零个或多个函数参数
template <typename T,typename...Args>
void foo(const T 6t,const Args6 ..  rest);

声明了foo是一个可变参数函数模板,它有一个名为T的类型参数,和一个名为Args的模板参数包。这个包表示零个或多个额外的类型参数。foo的函数参数列表包含一个const s类型的参数,指向T的类型,还包含一个名为rest的函数参数包,此包表示零个或多个函数参数。

与往常一样,编译器从函数的实参推断模板参数类型。对于一个可变参数模板,编译器还会推断包中参数的数目。例如,给定下面的调用

int i= 0;
double d=3.14;
string s="how now brown cow";
foo(i,s,42,d); //包中有三个参数
fog(s.42,"hi");//包中有两个参数
foo(d,s);      //包中有一个参数
foo("hi");      //空包

编译器会为foo实例化出四个不同的版本:

void foo(const int&,const string&,const int&,const double&);
void foo(const string&,const int&,const char[3]&);
void foo(const double&,const string&);
void foo(const char[3]&);

在每个实例中,T的类型都是从第一个实参的类型推断出来的。剩下的实参(如果有的话)提供函数额外实参的数目和类型。

2.2sizeof...运算符

当我们需要知道包中有多少元素时,可以使用sizeof...运算符。类似sizeof返回一个常量表达式

template<typename ...Args>
void g(Args .args)
{cout <<sizeof...(Args)<<end1;//类型参数的数目cout <<sizeof...(args)<<endl;//函数参数的数目
}

2.3可变参数模板的使用

void _ShowList()
{// 结束条件的函数std::cout << std::endl;
}template <class T, class... Args>
void _ShowList(T val, Args... args)
{std::cout << val << " ";_ShowList(args...);
}// args代表0-N的参数包
template <class... Args>
void CppPrint(Args... args)
{_ShowList(args...);
}
int main()
{CppPrint(1, 2, 2.2, string("xxxx"));
}

一般来说我们是使用递归的方式来将参数全部使用,当函数全部使用后就会匹配到结束函数。

template <class T>
void PrintArg(T t)
{std::cout << t << " ";
}
// args表示0-N的参数包
template <class... Args>
void CppPrintf(Args... args)
{int a[] = {0, (PrintArg(args), 0)...};cout << endl;
}

c++在编译时要确定数组a的大小来给空间,所以他会将里面的那个数据包展开,如图()中是一个逗号表达式,也就是有几个参数就会调用几下PrintArg。

2.4emplace_back()

emplace_back 是 C++ 中标准库容器 std::vector 的一个成员函数,用于在容器的尾部直接构造一个新元素,而不是先创建一个临时对象再拷贝或移动到容器中.

使用 emplace_back 可以直接在容器的尾部构造一个新元素,而不需要手动创建该元素的实例。emplace_back 接受任意数量的参数,这些参数会被传递给元素类型的构造函数,用于直接在容器中构造新元素。

所以网上有人说emplace_back代价更小,但是事实上移动拷贝代价更小,所以这句话应该有前提就是当元素类型是不可拷贝的时候。

在元素类型允许移动构造或移动赋值的情况下,emplace_backpush_back 的性能差异可能会减小甚至消失。

emplace_backpush_back 的主要性能差异在于:

  1. emplace_back 在容器中直接构造元素,避免了创建临时对象和拷贝/移动操作。
  2. push_back 在容器中插入一个已经构造的元素的拷贝或移动。

但是,如果元素类型具有移动语义(即具有移动构造函数和/或移动赋值运算符),那么在 push_back 中插入一个临时构造的元素,并在插入过程中执行移动操作,性能损失会相对较小。

因此,在元素类型允许移动拷贝时,emplace_backpush_back 的性能差异可能会减小,甚至没有明显的性能差异。在这种情况下,可以选择更符合语义的操作或更易读的代码。

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

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

相关文章

2024年2月份实时获取地图边界数据方法,省市区县街道多级联动【附实时geoJson数据下载】

首先&#xff0c;来看下效果图 在线体验地址&#xff1a;https://geojson.hxkj.vip&#xff0c;并提供实时geoJson数据文件下载 可下载的数据包含省级geojson行政边界数据、市级geojson行政边界数据、区/县级geojson行政边界数据、省市区县街道行政编码四级联动数据&#xff0…

Springboot+vue的大学生智能消费记账系统的设计与实现(有报告)。Javaee项目,springboot vue前后端分离项目

演示视频&#xff1a; Springbootvue的大学生智能消费记账系统的设计与实现&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot vue前后端分离项目 项目介绍&#xff1a; 本文设计了一个基于Springbootvue的大学生智能消费记账系统的设计与实现&#xff0c;采…

回溯题中借助哈希法来巧妙去重的操作

今天总结一下回溯法以来做过的这些题&#xff0c;我又发现一个困扰了我的问题&#xff0c;就是在491. 非递减子序列、46. 全排列、47. 全排列 II中都有涉及到用哈希法&#xff0c;去记录曾经用过的元素&#xff0c;下面来总结一下吧。 首先得知道&#xff0c;为什么会用到哈希法…

wordpress外贸成品网站模板

首页大图slider轮播&#xff0c;橙色风格的wordpress外贸网站模板 https://www.zhanyes.com/waimao/6250.html 蓝色经典风格的wordpress外贸建站模板 https://www.zhanyes.com/waimao/6263.html

RapidMiner数据挖掘2 —— 初识RapidMiner

本节由一系列练习与问题组成&#xff0c;这些练习与问题有助于理解多个基本概念。它侧重于各种特定步骤&#xff0c;以进行直接的探索性数据分析。因此&#xff0c;其主要目标是测试一些检查初步数据特征的方法。大多数练习都是关于图表技术&#xff0c;通常用于数据挖掘。 为此…

【复现】cellinx摄像设备 未授权漏洞_50

目录 一.概述 二 .漏洞影响 三.漏洞复现 1. 漏洞一&#xff1a; 四.修复建议&#xff1a; 五. 搜索语法&#xff1a; 六.免责声明 一.概述 cellinx是一家韩国的摄像设备 二 .漏洞影响 通过未授权访问可以创建用户进入后台&#xff0c;可能造成系统功能破坏。 三.漏洞复…

多线程面试题汇总

多线程面试题汇总 一、多线程1、线程的生命周期2、线程的创建&#xff08;函数创建&#xff09;3、线程的创建&#xff08;使用类&#xff09;4、守护线程 二、全局解释器锁1、使用单线程实现累加到5000000002、使用多线程实现累加到5000000003、总结 三、线程安全1、多线程之数…

SQL的1999语法

目录 交叉连接 实现交叉连接 自然连接 实现自然连接&#xff08;实际上就是内连接&#xff09; ON和USING 使用自然连接时要求两张表的字段名称相同&#xff0c;但是如果不相同或者两张表中有两组字段是重名,这时就要利用 ON 子句指定关联条件&#xff0c;利用 USING 子句…

【Android】使用Apktool反编译Apk文件

文章目录 1. 下载Apktool1.1 Apktool官网下载1.2 百度网盘下载 2. 安装Apktool3. 使用Apktool3.1 配置Java环境3.2 准备Apk文件3.3 反编译Apk文件3.3.1 解包Apk文件3.3.2 修改Apk文件3.3.3 打包Apk文件3.3.4 签名Apk文件 1. 下载Apktool 要使用Apktool&#xff0c;需要准备好 …

OpenSource - 一站式自动化运维及自动化部署平台

文章目录 orion-ops 是什么重构特性快速开始技术栈功能预览添砖加瓦License orion-ops 是什么 orion-ops 一站式自动化运维及自动化部署平台, 使用多环境的概念, 提供了机器管理、机器监控报警、Web终端、WebSftp、机器批量执行、机器批量上传、在线查看日志、定时调度任务、应…

2.14:二维数组、非函数实现strcat、strcmp、strcpy、strlen

1.编程实现二维数组的杨辉三角 程序代码&#xff1a; 1 #include<stdio.h>2 #include<string.h>3 #include<stdlib.h>4 int main(int argc, const char *argv[])5 {6 int n;7 printf("please enter n:");8 scanf("%d",&…

C++文件操作->文本文件(->写文件、读文件)、二进制文件(->写文件、读文件)

#include<iostream> using namespace std; #include <fstream>//头文件包含 //文本文件 写文件 void test01() { //1.包含头文件 fstream //2.创建流对象 ofstream ofs; //3.指定打开方式 ofs.open("test.txt", ios::out); //4.写…

蓝桥杯嵌入式学习记录——PWM输出

目录 一、PWM原理介绍 二、学习目的 三、cubeMX的配置 四、PWM输出代码 一、PWM原理介绍 PWM&#xff08;Pulse Width Modulation&#xff0c;脉宽调制&#xff09;是一种通过改变信号的脉冲宽度来控制电平的技术。它通过调整脉冲信号的占空比&#xff08;高电平时间与周期…

互联网加竞赛 基于计算机视觉的身份证识别系统

0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于机器视觉的身份证识别系统 该项目较为新颖&#xff0c;适合作为竞赛课题方向&#xff0c;学长非常推荐&#xff01; &#x1f9ff; 更多资料, 项目分享&#xff1a; https://gitee.com/dancheng-sen…

【STM32 CubeMX】I2C中断方式与DMA方式

文章目录 前言一、I2C中断方式1.1 CubeMX配置I2C中断1.2 I2C中断函数使用Master模式Mem模式 1.3 DMA方式发送和接收CubeMX配置IIC DMA方式Master模式Mem模式 总结 前言 在STM32 CubeMX环境中&#xff0c;I2C&#xff08;Inter-Integrated Circuit&#xff09;通信协议的实现可…

基于非线性系统的Lipschitz观测器simulink建模与仿真

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 4.1非线性系统及观测器概述 4.2 Lipschitz观测器原理 5.完整工程文件 1.课题概述 基于非线性系统的Lipschitz观测器simulink建模与仿真&#xff0c;该系统设计了一个观测器&#xff0c;称为Lipschitz观…

Capacity Maximization for Movable Antenna Enabled MIMO Communication

文章目录 II. SYSTEM MODEL AND PROBLEM FORMULATIONC. Problem Formulation III. PROPOSED ALGORITHMA. Alternating OptimizationB. Solution for Problem (P2-m) APPENDIX II. SYSTEM MODEL AND PROBLEM FORMULATION C. Problem Formulation 为了揭示支持 MA 的MIMO通信的…

【LeetCode: 103. 二叉树的锯齿形层序遍历 + BFS】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

嵌入式培训机构四个月实训课程笔记(完整版)-Linux ARM驱动编程第三天-ARM Linux ADC和触摸屏开发 (物联技术666)

链接&#xff1a;https://pan.baidu.com/s/1V0E9IHSoLbpiWJsncmFgdA?pwd1688 提取码&#xff1a;1688 教学内容&#xff1a; 1、ADC S3C2440的A/D转换器包含一个8通道的模拟输入转换器&#xff0c;可以将模拟输入信号转换成10位数字编码。 在A/D转换时钟频率为2.5MHz时&…

第三十回 张都监血溅鸳鸯楼 武行者夜走蜈蚣岭-python可接受任意数量参数的函数

武松回到孟州城&#xff0c;来到张都监后花园墙外&#xff0c;这是一个马院&#xff0c;问清楚后槽张团练他们三人还在鸳鸯楼吃酒&#xff0c;直接一刀杀了。武松从后门这里爬过墙&#xff0c;来到了厨房&#xff0c;将两个还在服侍的丫环杀了。 武松认得路&#xff0c;蹑手蹑…