C ++中的变量参数列表

时间:2011-04-16 19:24:05

标签: c++ variadic-functions printf

我有这个方法(使用来自vsnprintf手册页的代码):

MYSQL_RES *nsDatabase::queryf(const char *fmt,...){
  int n, size = 1024;
  char *query=NULL,*np;
  va_list ap;

  if (this->dbLink == NULL){
    return NULL;
  }

  query = (char *) malloc (size);
  if (query == NULL) return NULL;//memory error

  while (1) {
    va_start(ap, fmt);
    n=vsnprintf(query,size,fmt,ap);
    va_end(ap);
    if (n > -1 && n < size){// format string succeeded
      break;
    }
    if (n > -1)    /* glibc 2.1 */
       size = n+1; /* precisely what is needed */
    else           /* glibc 2.0 */
       size *= 2;  /* twice the old size */

    np = (char *) realloc (query, size);
    if (np == NULL) {
       printf("memory error\n");
       FREE(query);
       return NULL;//again memory error
    } else {
       query = np;
    }
  }

  MYSQL_RES *r = this->query(query);

  FREE(query);

  if(r == NULL){
    return NULL; //mysql error
  }

  return mysql_store_result(this->dbLink);
}

我想制作一般函数queryFormat,它将计算格式化字符串所需的大小,分配内存,打印格式并返回字符串(不是真实代码,伪代码) :

char *queryFormat(const char *fmt, va_list ap){
  // allocate memory
  // problem - can I use vsnprintf multiple times here (to determine the size of formatted string and allocate memory)
  return <formatted string>;
}

然后我需要从所有格式化查询的函数(而不是实际代码)中调用它:

queryRow(const char *fmt,...){
  va_start(ap, fmt);
  // I need to call vsnprintf many times in queryFormat ... Do I need to call va_start() before every call to fsnprintf ?
  char * formattedQuery = queryFormat(query, size, fmt, ap);
  va_end(ap);
  MYSQL_RES *r = this->query(formattedQuery);
  free(formattedQuery);

  ...

}

我在linux下为GCC编写,但代码应该可以在MinGW和CygWin下运行。

2 个答案:

答案 0 :(得分:0)

看到这一点 - http://publications.gbdirect.co.uk/c_book/chapter9/stdarg.html.There是Kernighan和Ritchie一书中的一个很好的例子。

修改的 它在C ++中也是一样的。你也可以看看这个。 - http://www.cplusplus.com/reference/clibrary/cstdarg/va_start/

编辑2 http://www.cplusplus.com/reference/clibrary/cstdio/vprintf/。我认为这会对你有帮助。

答案 1 :(得分:-1)

我相信这应该可以帮助任何搜索过这里的人找到解决问题的方法,如何重复使用&#34;函数中的va_list,它已作为参数传递:

// Incorrect:
/*void g(int foo, va_list ap) {
    vprintf("%s %s %s\n", ap);
    vprintf("%s %s %s\n", ap);
}*/

void g(int foo, va_list ap) {
    va_list copy;
    va_copy(copy, ap);
    vprintf("%s %s %s\n", copy);
    va_copy(copy, ap);
    vprintf("%s %s %s\n", copy);
}

void f(int foo, ...) {
    va_list ap;
    va_start(ap, foo);
    g(foo, ap);
    va_end(ap);
}

int main() {
    f(42, "aaa", "bbb", "ccc");
    return 0;
}