关于网友提出的“ C语言可变参数函数实现多级调用可变参数函数怎么弄。”问题疑问,本网通过在网上对“ C语言可变参数函数实现多级调用可变参数函数怎么弄。”有关的相关答案进行了整理,供用户进行参考,详细问题解答如下:
问题: C语言可变参数函数实现多级调用可变参数函数怎么弄。描述:
比如有一个func1(int a, char* fmt, ...);
有一个func2(int b, char* fmt, ...);
现在要在func1中调用func2要怎么做啊 ?
我像下面这样实现好像不行;
func1(int a, char* fmt, ...)
{
va_list args;
va_start(args, fmt);
func2(a, fmt, args);
va_end(args);
}
要怎么做啊?
解决方案1:
static func2(int b, char* fmt, ...);
func1(int a, char* fmt, ...);
func1(int a, char* fmt, ...)//如果在单文件内,没有在.h文件中包含,需要可调用函数在前面实现,或声明
{
va_list args;
va_start(args, fmt);
func2(a, fmt, args);
va_end(args);
}
NAME
printf, fprintf, sprintf, snprintf, vprintf, vfprintf, vsprintf, vsnprintf - formatted output conversion
SYNOPSIS
#include
int printf(const char *format, ...);
int fprintf(FILE *stream, const char *format, ...);
int sprintf(char *str, const char *format, ...);
int snprintf(char *str, size_t size, const char *format, ...);
#include
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);
#include
#include
#include
void func2(int n,const char *format,va_list args)
{
vprintf(format,args);
}
void func1(int n,const char *format,...)
{
va_list args;
va_start(args,format);
func2(n,format,args);
va_end(args);
}
int main()
{
func1(1,"%d %s\n",2,"hello");
return 0;
}
楼主如果能理解va_系列的源码可能更容易懂一些,va_list其实就是一个void *指针而已,它在va_start后被设置为参数栈上变量format之后4字节(因为const char*format)向sizeof(long)对齐的地址上。我们把这个地址传给func2,func2里调用vprintf系列函数,它们内部不再像printf那样操作自己的参数栈了,而是直接去这个va_list地址上开始往后找数据。
解决方案3:
仅供参考
#include解决方案4:
#include
#include
#ifdef WIN32
#include
#include
#else
#include
#include
#include
#define CRITICAL_SECTION pthread_mutex_t
#define _vsnprintf vsnprintf
#endif
//Log{
#define MAXLOGSIZE 100000000
#define ARRSIZE(x) (sizeof(x)/sizeof(x[0]))
#include
#include
char logfilename1[]="MyLog1.log";
char logfilename2[]="MyLog2.log";
char logstr[16000];
char datestr[16];
char timestr[16];
char ms10[3];
CRITICAL_SECTION cs_log;
FILE *flog;
int centisec() {
#ifdef WIN32
return ((GetTickCount()%1000L)/10)%100;
#else
struct timeval tv;
if (!gettimeofday(&tv,NULL)) {
return ((tv.tv_usec%1000000L)/10000)%100;
} else {
return 0;
}
#endif
}
#ifdef WIN32
void Lock(CRITICAL_SECTION *l) {
EnterCriticalSection(l);
}
void Unlock(CRITICAL_SECTION *l) {
LeaveCriticalSection(l);
}
#else
void Lock(CRITICAL_SECTION *l) {
pthread_mutex_lock(l);
}
void Unlock(CRITICAL_SECTION *l) {
pthread_mutex_unlock(l);
}
#endif
void LogV(const char *pszFmt,va_list argp) {
struct tm *now;
time_t aclock;
if (NULL==pszFmt||0==pszFmt[0]) return;
if (-1==_vsnprintf(logstr,ARRSIZE(logstr),pszFmt,argp)) logstr[ARRSIZE(logstr)-1]=0;
time(&aclock);
now=localtime(&aclock);
sprintf(datestr,"%04d-%02d-%02d",now->tm_year+1900,now->tm_mon+1,now->tm_mday);
sprintf(timestr,"%02d:%02d:%02d",now->tm_hour ,now->tm_min ,now->tm_sec );
sprintf(ms10,"%02d",centisec());
printf("%s %s.%s %s",datestr,timestr,ms10,logstr);
flog=fopen(logfilename1,"a");
if (NULL!=flog) {
fprintf(flog,"%s %s.%s %s",datestr,timestr,ms10,logstr);
if (ftell(flog)>MAXLOGSIZE) {
fclose(flog);
if (rename(logfilename1,logfilename2)) {
remove(logfilename2);
rename(logfilename1,logfilename2);
}
flog=fopen(logfilename1,"a");
if (NULL==flog) return;
}
fclose(flog);
}
}
void Log(const char *pszFmt,...) {
va_list argp;
Lock(&cs_log);
va_start(argp,pszFmt);
LogV(pszFmt,argp);
va_end(argp);
Unlock(&cs_log);
}
//Log}
int main(int argc,char * argv[]) {
int i;
#ifdef WIN32
InitializeCriticalSection(&cs_log);
#else
pthread_mutex_init(&cs_log,NULL);
#endif
for (i=0;i<10000;i++) {
Log("This is a Log %04d from FILE:%s LINE:%d\n",i, __FILE__, __LINE__);
}
#ifdef WIN32
DeleteCriticalSection(&cs_log);
#else
pthread_mutex_destroy(&cs_log);
#endif
return 0;
}
func2一般参数用va_list