C libcurl获取输出到字符串中


94

我想将此curl函数的结果存储在变量中,该怎么办?

#include <stdio.h>
#include <curl/curl.h>

int main(void)
{
  CURL *curl;
  CURLcode res;

  curl = curl_easy_init();
  if(curl) {
    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    res = curl_easy_perform(curl);

    /* always cleanup */
    curl_easy_cleanup(curl);
  }
  return 0;
}

谢谢,我这样解决了它:

#include <stdio.h>
#include <stdlib.h>
#include <curl/curl.h>

function_pt(void *ptr, size_t size, size_t nmemb, void *stream){
    printf("%d", atoi(ptr));
}

int main(void)
{
  CURL *curl;
  curl = curl_easy_init();
  if(curl) {
    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, function_pt);
    curl_easy_perform(curl);
    curl_easy_cleanup(curl);
  }
  system("pause");
  return 0;
}

1
只是要指出在function_pt()中的解决方案中,您正在将ptr中的字符串转换为整数,以便在输出中将其转换回字符串。您可以直接输出字符串(并查看完整响应)。
zzz 2015年

2
这是指向cURL示例的链接 curl.haxx.se/libcurl/c/getinmemory.html
lafferc 2015年

1
CURLcode res;未使用
fnc12

相同的问题,但对于C ++而不是c,请转至此处:将cURL内容结果保存到C ++中的字符串中
Trevor Boyd Smith

Answers:


114

您可以设置一个回调函数来使用以下方法接收传入的数据块 curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, myfunc);

回调将使用一个用户定义的参数,您可以使用 curl_easy_setopt(curl, CURLOPT_WRITEDATA, p)

这是一段代码,该代码片段将缓冲区传递struct string {*ptr; len}给回调函数,并在每次调用时使用realloc()增大该缓冲区。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <curl/curl.h>

struct string {
  char *ptr;
  size_t len;
};

void init_string(struct string *s) {
  s->len = 0;
  s->ptr = malloc(s->len+1);
  if (s->ptr == NULL) {
    fprintf(stderr, "malloc() failed\n");
    exit(EXIT_FAILURE);
  }
  s->ptr[0] = '\0';
}

size_t writefunc(void *ptr, size_t size, size_t nmemb, struct string *s)
{
  size_t new_len = s->len + size*nmemb;
  s->ptr = realloc(s->ptr, new_len+1);
  if (s->ptr == NULL) {
    fprintf(stderr, "realloc() failed\n");
    exit(EXIT_FAILURE);
  }
  memcpy(s->ptr+s->len, ptr, size*nmemb);
  s->ptr[new_len] = '\0';
  s->len = new_len;

  return size*nmemb;
}

int main(void)
{
  CURL *curl;
  CURLcode res;

  curl = curl_easy_init();
  if(curl) {
    struct string s;
    init_string(&s);

    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, writefunc);
    curl_easy_setopt(curl, CURLOPT_WRITEDATA, &s);
    res = curl_easy_perform(curl);

    printf("%s\n", s.ptr);
    free(s.ptr);

    /* always cleanup */
    curl_easy_cleanup(curl);
  }
  return 0;
}

1
真好 如果将所有这些声明size_tlen本身除外)都更好const
2014年

1
对于C ++,std::string请转到此处
Trevor Boyd Smith,

33

以下答案是C ++的实现方式,它使用std::string,而不是以null终止的字符串。它仍然使用回调函数(无法解决),但也使用try / catch处理分配错误。

#include <iostream>
#include <string>
#include <curl/curl.h>

size_t CurlWrite_CallbackFunc_StdString(void *contents, size_t size, size_t nmemb, std::string *s)
{
    size_t newLength = size*nmemb;
    try
    {
        s->append((char*)contents, newLength);
    }
    catch(std::bad_alloc &e)
    {
        //handle memory problem
        return 0;
    }
    return newLength;
}
int main()
{
    CURL *curl;
    CURLcode res;

    curl_global_init(CURL_GLOBAL_DEFAULT);

    curl = curl_easy_init();
    std::string s;
    if(curl)
    {

        curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");

        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); //only for https
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); //only for https
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, CurlWrite_CallbackFunc_StdString);
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, &s);
        curl_easy_setopt (curl, CURLOPT_VERBOSE, 1L); //remove this to disable verbose output


        /* Perform the request, res will get the return code */
        res = curl_easy_perform(curl);
        /* Check for errors */
        if(res != CURLE_OK)
        {
            fprintf(stderr, "curl_easy_perform() failed: %s\n",
                    curl_easy_strerror(res));
        }

        /* always cleanup */
        curl_easy_cleanup(curl);
    }

    std::cout<<s<<std::endl;

    std::cout<< "Program finished!" << std::endl;
}

我认为std :: string :: append可以使该回调函数更简单。
rnickb

@rnickb你是对的;s->append((char*)contents. nmemb);与我完美协作,更加简洁。另外,回调的正式函数签名具有char*第一个参数,因此您可以使用它并省略强制转换。最后,s->resize()实际上是初始化新分配的空间。无论如何,您都将要覆盖它,s->reserve()这样会更有效率。
Jeinzi '19

这对我很有帮助。您还可以举一个使用HTTP POST的示例吗:-)
Wolfenstein勋爵

9

从这里阅读手册:http : //curl.haxx.se/libcurl/c/curl_easy_setopt.html我认为您需要多次调用CURL_SETOPT,第一个是要处理的URL,第二个是类似的:

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, function_ptr);

其中function_ptr与此签名匹配:

size_t function( void *ptr, size_t size, size_t nmemb, void *stream)

这里发生的是您表示一个回调函数,当libcurl有一些输出要从调用的任何传输写入时,libcurl就会调用该函数。您可以获取它以自动写入文件,或向其传递指向将处理输出本身的函数的指针。使用此功能,您应该能够将各种输出字符串组装成一个片段,然后在程序中使用它们。

我不确定您可能还需要设置哪些其他选项/哪些因素会影响您希望应用程序的运行方式,因此请在该页面中仔细查看。


0

这是alex-jasmin接受的答案的C ++风格

#include <iostream>
#include <string>
#include <curl/curl.h>

size_t writefunc(void *ptr, size_t size, size_t nmemb, std::string *s) 
{
  s->append(static_cast<char *>(ptr), size*nmemb);
  return size*nmemb;
}

int main(void)
{
  CURL *curl = curl_easy_init();
  if (curl)
  {
    std::string s;

    curl_easy_setopt(curl, CURLOPT_URL, "curl.haxx.se");
    curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, writefunc);
    curl_easy_setopt(curl, CURLOPT_WRITEDATA, &s);

    CURLcode res = curl_easy_perform(curl);

    std::cout << s << std::endl;

    /* always cleanup */
    curl_easy_cleanup(curl);
  }
  return 0;
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.