C/C++不定参数的使用

文章目录

  • C语言的不定参
  • C++的不定参


C语言的不定参

C语言的不定参数最常见的应用示例就是printf函数,如下,参数列表中的...表示不定参数列表

#include <stdio.h>
int printf(const char *format, ...);

试着模拟实现C语言的printf函数

void myprintf(const char *fmt, ...)
{//TODO
}

C语言中,对于...不定参列表,要用va_*系列宏函数操作

#include <stdarg.h>void va_start(va_list ap, last);
type va_arg(va_list ap, type);
void va_end(va_list ap);
void va_copy(va_list dest, va_list src);
  • va_list

    va_list可以理解为一个指针类型,开始时,通过调用va_start,它会指向不定参列表的第一个参数

  • va_start

    我们知道,C/C++的函数参数通过压栈的方式传入,不定参列表的多个参数也需要压栈。

    在这里插入图片描述

    va_start的作用是使ap指向不定参列表的第一个参数。通过不定参数列表前的最后一个函数参数,找到不定参列表的首元素,并使ap(va_list类型的指针)指向首元素。因此,C函数使用...不定参数,前面至少包含一个其它参数,由它提供不定参数的起始位置。

  • va_arg

    va_arg的作用是在不定参数列表中,从ap指向的参数开始,逐个返回type类型的数据,一般需要循环调用va_arg,在此过程中ap会不断往后走

  • va_end

    ap使用结束后就失效了,为了避免野指针的引用,va_end可以销毁ap指针

  • va_copy

    拷贝src到dest,使得二者指向同一个不定参数列表,src不用再次调用va_start,但要调用va_end

Test Demo:

void myprintf(size_t num, ...)
{va_list ap1;va_start(ap1, num);va_list ap2;va_copy(ap2, ap1);for (int i = 0; i < num; i++){char ch = va_arg(ap1, int);std::cout << ch;}std::cout << std::endl;for (int i = 0; i < num; i++){char ch = va_arg(ap2, int);std::cout << ch;}std::cout << std::endl;va_end(ap1);va_end(ap2);
}int main()
{myprintf(3, 'L', 'O', 'L');return 0;
}

⭕输出结果

[ckf@VM-8-12-centos 1.before]$ ./vargs 
LOL
LOL

注意:va_arg对于可变参数的处理是有一些规则的,特别是对于小整数类型(如charfloat)。这些类型在传递给va_arg时会被默认转换为intdouble,而不是它们的实际类型。因此,在Test Demo中,对于传入的char类型参数,在调用va_arg时也应该指定type为int,获得返回值后再转换为实际类型char

以下函数可以对不定参数列表进行格式化操作。

#include <stdarg.h>int vprintf(const char *format, va_list ap);
int vfprintf(FILE *stream, const char *format, va_list ap);
int vsprintf(char *str, const char *format, va_list ap);
int vsnprintf(char *str, size_t size, const char *format, va_list ap);

它们的作用与printf相同,都可以通过参数format,对不定参数进行格式化。区别在于:printf使用不定参...作为格式化的数据,而上面以v开头printf结尾的函数,使用ap指向的不定参数列表作为格式化的数据。除此之外,内部的区别就在于输出的方向了,vprintf将格式化后的字符串输出到stdout,vfprintf输出到指定的文件流,vsprintfvsprintf输出到指定的内存空间str,且后者可以指定输出的字节个数。

Test Demo:

void myprintf(const char *fmt, ...)
{va_list ap;va_start(ap, fmt);char str[64] = {0};int len = vsnprintf(str, sizeof(str), fmt, ap);str[len] = 0;std::cout << str << std::endl;va_end(ap);
}
int main()
{myprintf("%s %c %d", "你好", '!', 2024);return 0;
}

⭕输出结果

[ckf@VM-8-12-centos 1.before]$ ./vargs 
你好 ! 2024

C++的不定参

💭补充:

  1. args参数包作为参数传递给另一个函数时,记得加...,表示这个args参数是不定参数包
  2. 函数接收Args...类型的参数包时,通常可以将参数设置为&&万能引用,减少拷贝的成本
  3. C++11标准库中,很多构造方法都使用了以不定参数包为参数的条目,方便一些无法缺点参数个数的实例的构造

例如:

  • std::make_shared,由于并不确定构造智能指针指向的对象类型需要传入多少参数,C++11使用了Args不定参数包
    在这里插入图片描述

  • emplace函数,C++11很多容器都重载了这个函数,emplace支持可变参数包,并且用参数包中的参数作为新插入元素构造函数的参数。

    在这里插入图片描述

Test Demo

template <class T>
void myprintfcpp(T val) // 终止函数
{std::cout << val;
}template <class T, class... Args>
void myprintfcpp(T val, Args &&...args)
{std::cout << val;myprintfcpp(std::forward<Args>(args)...);
}int main()
{myprintfcpp(3, 'L', 'O', 'L');std::cout << std::endl;return 0;
}

⭕输出结果

[ckf@VM-8-12-centos 1.before]$ ./vargs 
3LOL

END…

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

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

相关文章

Android:BackStackRecord

BackStackRecord:fragment回退栈,继承自FragmentTransaction,并且实现了OpGenerator接口,OpGenerator接口用来添加或弹出事务的,后面会提到。 从《Android:从源码看FragmentManager如何工作》文章知道,每次beginTransaction会创建一个BackStackRecord对象,改对象持有f…

C++基础——文件操作

文章目录 1 概述2 文本文件2.1 写文件2.1.1 写文件流程2.1.2 文件打开方式 2.2 读文件 3 二进制文件3.1 写文件3.2 读文件 1 概述 程序最基本的操作之一就是文件操作&#xff0c;程序运行时的数据都是临时数据&#xff0c;当程序结束后就不复存在了。通常都是通过文件或其他持…

【vue实战项目】通用管理系统:信息列表,信息录入

本文为博主的vue实战小项目系列中的第六篇&#xff0c;很适合后端或者才入门的小伙伴看&#xff0c;一个前端项目从0到1的保姆级教学。前面的内容&#xff1a; 【vue实战项目】通用管理系统&#xff1a;登录页-CSDN博客 【vue实战项目】通用管理系统&#xff1a;封装token操作…

深入理解HTTP协议

一、简介 超文本传输协议&#xff08;英文&#xff1a;HyperText Transfer Protocol&#xff0c;缩写&#xff1a;HTTP&#xff09;是一种用于分布式、协作式和超媒体信息系统的应用层协议。HTTP是万维网的数据通信的基础。 二、协议概述 HTTP是一个客户端终端&#xff08;用…

栈模拟先序后序中序遍历(非递归遍历)

先序遍历&#xff1a; vector<int> preorderTraversal(TreeNode* u) {stack<TreeNode*>stk;vector<int>res;if(unullptr) return res;while(stk.size()||u){if(u){res.push_back(u->val);//遍历当前结点stk.push(u);//记录当前递归层uu->left;//遍历左…

显示Excel功能区或工具栏的方法不少,其中快捷方式最快

Microsoft Excel是Office套件中最复杂的工具之一&#xff0c;它提供了大量功能&#xff0c;其中大部分都是使用工具栏操作的。缺少工具栏使Excel很难完成工作。 如果Excel中没有这些关键元素&#xff0c;你将无法快速完成工作&#xff0c;因此&#xff0c;可以理解的是&#x…

处理机调度与作业调度

处理机调度 一个批处理型作业&#xff0c;从进入系统并驻留在外存的后备队列上开始&#xff0c;直至作业运行完毕&#xff0c;可能要经历如下的三级调度 高级调度 也称为作业调度、长程调度、接纳调度。调度对象是作业 主要功能&#xff1a; 挑选若干作业进入内存 为作业创建…

flutter Running Gradle task ‘assembleDebug‘

flutter Running Gradle task assembleDebug Running Gradle task assembleDebug新问题描述新问题解决方案Running Gradle task ‘assembleDebug’ 用Android Stduio创建Flutter项目的时候,会出现各种问题,踩了一个又一个,最后编译的时候可能会出现一直显示Running Gradle …

在Pycharm中创建项目新环境,安装Pytorch

在python项目中&#xff0c;很多项目使用的各类包的版本是不一致的。所以我们可以对每个项目有专属于它的环境。所以这个文章就是教你如何创建新环境。 一、创建新环境 首先我们需要去官网下载conda。然后在Pycharm下面添加conda的可执行文件。 用conda创建新环境。 二、…

多要素气象环境监测站知识科普

随着工业化和城市化的快速发展&#xff0c;气象环境的影响越来越受到人们的关注。为了更好地保护我们的环境&#xff0c;一款WX-CQ12 多要素气象环境监测站应运而生。这款监测站可以全方位地监测气象环境中的温度、湿度、气压、风速、风向、雨量、太阳辐射等重要要素&#xff0…

Python实现的顺序查找算法

一、算法过程描述&#xff1a; 最基本的查找技术的过程&#xff1a;从表中的第一个记录开始&#xff0c;逐个进行记录的 关键字和给定值比较&#xff0c;若某个记录的关键字和给定值相等&#xff0c;则查找成功&#xff0c;找到所查的记录&#xff1b; 如果直到最后一个记录&am…

vue3高德地图使用,地址搜索,地址逆解析

在vue3项目里使用高德地图 高德地图文档 先在项目的index.html页面里添加一些东西 <script type"text/javascript">window._AMapSecurityConfig {securityJsCode: "xxxxxxxxxxxxx", //高德安全码};</script> <script src"https://…

Python---文件

文件--- 内存中存放的数据在计算机关机后就会消失。要长久保存数据&#xff0c;就要使用硬盘、光盘、U 盘等设备。为了便于数据的管理和检索&#xff0c;引入了“文件”的概念。 一篇文章、一段视频、一个可执行程序&#xff0c;都可以被保存为一个文件&#xff0c;并赋予一个…

@FeignClient本地项目启动调用测试服务地址,并兼容发布测试环境

1.FeignClient对应api接口的使用如下&#xff1a; /*** author ys* description 对外服务接口*/ FeignClient(name "xx-shop", path "/shop", url "${api-third.xx-shop:}") public interface ExterlControllerApi {/*** 获取所有的类型** p…

使用docker-compose优雅部署rocketMQ

使用docker-compose优雅部署RocketMQ 随着市场的发展&#xff0c;越来越多的复杂场景出现在我们日常的开发工作中。随之也越来越多的好的工具&#xff0c;也同步出现在程序员的学习范围清单内。好的工具提高产品性能的同时&#xff0c;也带来了很多安装上的问题&#xff0c;do…

小米的算法部署岗对新手是真的友好

大家好啊&#xff0c;我是董董灿。 自从开始写一些AI行业的岗位介绍&#xff0c;就养成了一个习惯&#xff0c;在上下班的路上经常就会打开某聘瞧一瞧。 导致之前一年不看的某聘认为我要看机会换工作&#xff0c;疯狂给我推猎头&#xff0c;然后电话就进来了。 不堪骚扰的我…

【数据结构】源码角度剖析PriorityQueue

目录 认识 Queue 认识 PriorityQueue PriorityQueue为什么要用二叉堆&#xff1f; PriorityQueue构造方法源码分析 PriorityQueue 的属性 构造方法 JDK1.8传入不可比较的对象 JDK17传入不可比较的对象 传入带有Collection接口的对象 instanceof 关键字 Offer方法分析…

.net core 连接数据库,通过数据库生成Modell

1、安装EF Core Power Tools&#xff1a;打开Vs开发工具→扩展→管理扩展 2、(切记执行这步之前确保自己的代码不存在编写或者编译错误&#xff01;)安装完成后在你需要创建数据库实体的项目文件夹上面单击右键&#xff0c;找到EF Core 工具&#xff08;必须安装扩展之和才会有…

(Ant X6)子组件里的流程图画布无法显示

(Ant X6)子组件里的流程图画布无法显示 问题背景&#xff1a;侧导航页面都是子组件,建模页面的画布无法显示 解决前&#xff1a; 解决后&#xff1a; 解决思路&#xff1a;点击建模菜单时再次加载对应组件 在 Vue 中&#xff0c;每个组件都有一个唯一的 key 属性。当组件的 ke…

vue升级题

不熟悉的&#xff1a; 2&#xff0c; 3.你用过befcoreDetory 吗&#xff1f;清除定时器&#xff0c;第一个和第二个再看一下 实例加载完成是在哪个生命周期--beforecreate 7.父子组件生命周期执行顺序&#xff1f;为什么这么渲染&#xff1f;场景 8.简单描述每个周期具体适…