免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1801 | 回复: 0
打印 上一主题 下一主题

linux va_start va_arg va_end原理及实例 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2009-02-19 11:44 |只看该作者 |倒序浏览

  最近一直使用sprintf函数,感慨于内面的不定参数,闲来想研究研究.sprintf(a,)
1:当无法列出传递函数的所有实参的类型和数目时,可用省略号指定参数表
void foo(...);
void foo(parm_list,...);
2:函数参数的传递原理
函数参数是以数据结构:栈的形式存取,从右至左入栈.
3.linux下改写man的一个例子:

               
               
                #include stdio.h>
#include stdarg.h>
int foo(char* fmt,...)
{
va_list ap;
int d;
char c, *s;
va_start(ap, fmt);
while (*fmt)
  switch(*fmt++)
  {
   case 's':/* string */
    s = va_arg(ap, char *);
    printf("string %s\n", s);
    break;
   case 'd':/* int */
    d = va_arg(ap, int);
    printf("int %d\n", d);
    break;
   case 'c':/* char */
    /* need a cast here since va_arg only
      takes fully promoted types */
    c = (char) va_arg(ap, int);
    printf("char %c\n", c);
    break;
   }
  va_end(ap);
  return 0;
}
int main()
{
  char* a = "aaa";
  int b = 65;
  char c = 'c';
  foo("%s,%d,%c",a,b,c);     
  return 0;
}
   其实大家可以看出和sprintf用法差不多....man里面还有个va_copy我这里就不写了,个人觉得用处不大.
4.自己实现个简单的va系列函数.这个程序只支持int输出...别整个char*了说为什么不对.与我而言是不求甚解的.
#include stdio.h>
#include stdlib.h>
void myprintf(char* fmt, ...)        //一个简单的类似于printf的实现,//参数必须都是int 类型
{
    char* pArg=NULL;               //等价于原来的va_list
    char c;
  
    pArg = (char*) &fmt;          //注意不要写成p = fmt !!因为这里要对参数取址,而不是取值
    pArg += sizeof(fmt);         //等价于原来的va_start        

    do
    {
        c =*fmt;
        if (c != '%')
        {
            putchar(c);            //照原样输出字符
        }
        else
        {
           //按格式字符输出数据
           switch(*++fmt)
           {
            case'd':
                printf("%d",*((int*)pArg));         
                break;
            case'x':
                printf("%#x",*((int*)pArg));
                break;
            default:
                break;
            }
            pArg += sizeof(int);               //等价于原来的va_arg
        }
        ++fmt;
    }while (*fmt != '\0');
    pArg = NULL;                               //等价于va_end
    return;
}
int main(int argc, char* argv[])
{
    int i = 1234;
    int j = 5678;
  
    myprintf("the first test:i=%d\n",i,j);
    myprintf("the secend test:i=%d; %x;j=%d;\n",i,0xabcd,j);
    system("pause");
    return 0;
}
好了,这篇over~~~~


本文来自ChinaUnix博客,如果查看原文请点:http://blog.chinaunix.net/u2/76292/showart_1833858.html
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP