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

2
ось посилання на приклад cURL curl.haxx.se/libcurl/c/getinmemory.html
lafferc

1
CURLcode res;не використовується
fnc12

те саме питання, але для C ++ замість c перейдіть сюди: Збережіть результат вмісту CURL у рядок на C ++
Тревор Бойд Сміт

Відповіді:


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_t(крім lenсебе) будуть оголошені const.
alk

1
для C ++ std::stringйти тут
Trevor Boyd Smith

33

Наступна відповідь - це спосіб зробити це на C ++ std::string, замість рядка, що закінчується нулем. Він все ще використовує функцію зворотного виклику (її неможливо обійти), але також обробляє помилку розподілу за допомогою функції 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 може значно спростити цю функцію зворотного виклику.
Ryan Burn

@rnickb Ти маєш рацію; s->append((char*)contents. nmemb);працює зі мною бездоганно і більш лаконічно. Крім того, офіційний підпис функції для зворотного виклику має char*перший аргумент, тому ви можете використовувати його та пропустити кастинг. І нарешті, s->resize()фактично ініціалізує щойно виділений простір. Оскільки ви все одно збираєтесь його перезаписати, це s->reserve()було б ефективніше.
Jeinzi

Це мені дуже допомогло. Ви також можете навести приклад того, як це зробити за допомогою HTTP POST, будь ласка :-)
лорд Вольфенштайн

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 буде викликати, коли вона має певний результат для запису із будь-якої передачі, яку ви викликали. Ви можете змусити його автоматично писати у файл або передати вказівник на функцію, яка обробляє сам вивід. Використовуючи цю функцію, ви зможете зібрати різні вихідні рядки в один шматок, а потім використовувати їх у своїй програмі.

Я не впевнений, які ще варіанти вам потрібно встановити / що ще впливає на те, як ви хочете, щоб ваш додаток поводився, тож добре перегляньте цю сторінку.


0

Ось смак 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;
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.