以下程序主要包括三个主要函数:

  • 一个最简单的可变形参函数实例;
  • 一个简单的printf功能的实例;
  • 一个打印字符串函数(辅助);

其中myPrintf函数,实现了printf的部分简单功能,并没有去实现格式化输出操作(%2d %.2f 等),

因为只是它一个实例而已,目的是更好理解掌握可变参数。

后续有兴趣时,我会继续完善的。

/*******************************************************************************************************
* 名 称: main.c
* 功 能:C语言可变形参 variable arguments,典型案例是printf和scanf
实现了一个简单的可变参数函数例子 ShowArgs
实现了一个简单printf功能的函数 myPrintf
* 描 述:可变参数的个数是不确定的
可变参数的类型是不确定的
printf和scanf在实现时,通过前面的%d %f 等获取可变参数的个数和类型
可变参数由定义在stdarg.h头文件中的宏:va_list va_start va_arg va_end等组成
va_list:用于保存参数列表
va_start(v,l):将参数列表的首地址 l 赋值给 v
va_arg(v,l):获取参数列表中类型为 l 的参数的值,并将参数列表指针向后移动到下一个参数处
va_end(v):结束可变参数
* 作 者:JarvisChu
* 时 间:2011-7-17 创建
********************************************************************************************************/
#include <stdio.h>
#include <stdarg.h>
#include <string.h> //在此处选择函数
//#define SHOW_ARGS
//#define MY_PRINTF
#define SHOW_STRING_BY_CHAR //一个简单的实例,num代表可变参数的个数,类型全部默认为double型
void ShowArgs(int num,...){
int i=;
double var;
va_list ls;
va_start (ls,num); //获取可变参数列表的首地址,赋值给ls
printf("-----------%d个可变参数,如下:----------\n",num);
while(i<num){
var = va_arg(ls,double);//返回类型为double的的可变参数值,并使ls指向下一个可变参数
printf("参数%d:%f\n",++i,var);
}
va_end(ls); //结束
printf("\n");
} //实现类似printf的部分功能,利用putchar向屏幕输出字符
void myPrintf(char* str,...){
int i, len;
// int num = 0; //num为可变参数个数 char pre,cur; //前一个字符和当前的字符 va_list ls;
va_start(ls,str); len = strlen(str); //获取字符串长度 pre = ' '; //初始化时,pre为空格(只要不是特殊字符或转义字符就行) for(i=;i<len;i++){
cur = str[i];
switch(pre){
case '%': //前一个字符值是%字符,我们就要考虑%d %f 这种情况了
switch(cur){
case 'd': //%d,输出整数
printf("%d",va_arg(ls,int));//将可变部分的int类型变量替代%d输出,本来很不想在这里用printf的,
break; //但是没有想到其他的办法向屏幕输出一个整型,姑且用printf吧
case 'f': //%f,输出float double类型
printf("%f",va_arg(ls,double));
break;
case 'c': //%c,输出一个字符
putchar(cur);
break;
}
break;
default: //前一个字符不是什么特殊字符
if(cur != '%'){ //当前字符不是%
putchar(cur);
}
break;
}
pre = cur;
}
va_end(ls);
} //辅助函数:将一个字符串,一个字符一个字符的输出
void ShowStringByChar(char* str){
int i, len;
char ch;
len = strlen(str);
for(i=;i<len;i++){
ch = str[i];
putchar(ch);
printf("\n----\n");
}
}
int main()
{
#ifdef SHOW_ARGS
ShowArgs();
ShowArgs(,1.1);
ShowArgs(,1.1,2.2);
#endif #ifdef MY_PRINTF
myPrintf("Jarvis");
myPrintf("\n"); //注意\n \t为一个字符,不要理解成了char[0] = '\' ; char[1] = 'n';
myPrintf("\n");
myPrintf("Jarvis\n");
myPrintf("Jarvis %d\n",);
myPrintf("Jarvis %d\t%d\n",, );
myPrintf("Jarvis %d\t%d\t%f\n",, ,63.2);
#endif #ifdef SHOW_STRING_BY_CHAR
ShowStringByChar("123\t456\n789%d");
# endif
return ;
}

C语言中可变形参简单实例的更多相关文章

  1. C语言中可变参数的函数(三个点,“...”)

    C语言中可变参数的函数(三个点,“...”) 本文主要介绍va_start和va_end的使用及原理. 在以前的一篇帖子Format MessageBox 详解中曾使用到va_start和va_end ...

  2. C语言中可变参数的使用

    在C语言程序编写中我们使用最多的函数一定包括printf以及很多类似的变形体.这个函数包含在C库函数中,定义为 int printf( const char* format, ...); 除了一个格式 ...

  3. c语言中函数的简单介绍

    c语言中函数的介绍: 函数,简单的说就是代码的打包.存放在一个地方,当需要的时候调用. 函数分类: 1.无参无返回值函数 void func() 2.无参有返回值函数  int func() 3.有参 ...

  4. C语言中可变参数的原理——printf()函数

    函数原型: int printf(const char *format[,argument]...) 返 回 值: 成功则返回实际输出的字符数,失败返回-1. 函数说明: 使用过C语言的人所再熟悉不过 ...

  5. C语言中可变参数函数实现原理

    C函数调用的栈结构 可变参数函数的实现与函数调用的栈结构密切相关,正常情况下C的函数参数入栈规则为__stdcall, 它是从右到左的,即函数中的最右边的参数最先入栈.例如,对于函数: void fu ...

  6. C语言中可变参数的用法

    原文地址: http://blog.csdn.net/wooin/archive/2006/04/29/697106.aspx   我们在C语言编程中会遇到一些参数个数可变的函数,例如printf() ...

  7. java中的反射简单实例

    package club.reflection.entity.User; /** * 实体类 * */ public class User { public String name; private ...

  8. c 中可变参数的实现

    我们在C语言编程中有时会遇到一些参数个数可变的函数,例如printf()函数,其函数原型为:     例一: int   printf(   const   char*   format,   ... ...

  9. c语言中函数参数入栈的顺序是什么?为什么

    看到面试题C语言中函数参数的入栈顺序如何? 自己不知道,边上网找资料.下面是详细解释 #include <stdio.h> void foo(int x, int y, int z){   ...

随机推荐

  1. 【小米oj】找出单独的数字

    题目链接:https://code.mi.com/problem/list/view?id=2&cid=0&sid=26251#codearea 描述 给出N个数字.其中仅有一个数字出 ...

  2. 【nyoj-1233】差值

    描述 输入一个整数数组,将它们连接起来排成一个数,找出能排出的所有数字中最大,最小的两个,输出两个数的差值.例如输入数组{1,  2},则输出9.     输入 第一行输入一个整数T,表示有T组测试数 ...

  3. 浅谈《剑指offer》原题:不使用条件、循环语句求1+2+……+n

    转载自:浅谈<剑指offer>原题:求1+2+--+n 如侵犯您的版权,请联系:windeal12@qq.com <剑指offer>上的一道原题,求1+2+--+n,要求不能使 ...

  4. DR模式下的高可用的LVS(LVS+keepalived)

    一.keepalived 在DR模式下,使用Keepalived实现LVS的高可用.Keepalived的作用是检测服务器的状态,如果有一台web服务器 宕机,或工作出现故障,Keepalived将检 ...

  5. c# 实体处理工具类

    using System; using System.Collections; using System.Collections.Generic; using System.ComponentMode ...

  6. Kotlin Reference (八) Classes and Objects

    most from reference 类 Kotlin的类的声明使用关键字class class Invoice { } 类声明由类名.类头(指定其类型参数,构造函数等)和类体组成,由大括号括起来. ...

  7. 数据库连接池----Druid配置详解

    什么是连接池? 数据库连接池出现的原因在数据库连接资源的低效管理,使用数据库连接池是基于设计模式中的资源池的概念,从而解决资源频繁是分配.释放所造成的问题. 数据库连接池的基本思想就是为数据库连接建立 ...

  8. Linux系统中的一些重要的目录

    Linux系统中的一些重要的目录 1)bin目录 :基础系统所需要的命令位于此目录,是最小系统所需要的命令,如:ls, cp, mkdir等.这个目录中的文件都是可执行的,一般的用户都可以使用2)sb ...

  9. UDP示例

    android学习笔记18--------------UDP示例 分类: android2011-11-10 10:07 848人阅读 评论(0) 收藏 举报 androidbufferexcepti ...

  10. 【解题报告】CF Round #320 (Div. 2)

    Raising Bacteria 题意:盒子里面的细菌每天会数量翻倍,你可以在任意一天放任意多的细菌,最后要使得某天盒子里面的细菌数量等于x,求至少要放多少个细菌 思路:显然,翻倍即为二进制左移一位, ...