C libcurl ottiene l'output in una stringa


94

Voglio memorizzare il risultato di questa funzione curl in una variabile, come posso farlo?

#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;
}

grazie, l'ho risolto così:

#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
Solo per indicare nella tua soluzione in function_pt () stai convertendo la stringa in ptr in intero per riconvertirla in stringa nell'output. È possibile visualizzare la stringa direttamente (e vedere la risposta completa).
zzz

2
ecco un collegamento all'esempio cURL curl.haxx.se/libcurl/c/getinmemory.html
lafferc

1
CURLcode res;non è utilizzato
fnc12

stessa domanda ma per C ++ invece di c vai qui: Salva il risultato del contenuto cURL in una stringa in C ++
Trevor Boyd Smith

Risposte:


114

È possibile impostare una funzione di richiamata per ricevere blocchi di dati in arrivo utilizzando curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, myfunc);

Il callback prenderà un argomento definito dall'utente che puoi impostare usando curl_easy_setopt(curl, CURLOPT_WRITEDATA, p)

Ecco uno snippet di codice che passa un buffer struct string {*ptr; len}alla funzione di callback e aumenta quel buffer a ogni chiamata utilizzando 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
Bello. Ancora più bello se tutti quelli size_t(oltre a lense stesso) fossero dichiarati const.
alk

1
per C ++ std::stringvai qui
Trevor Boyd Smith

34

La seguente risposta è il modo C ++ per farlo, con std::string, invece di una stringa con terminazione null. Usa ancora una funzione di callback (non c'è modo di aggirarla), ma gestisce anche l'errore di allocazione usando 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;
}

Penso che std :: string :: append possa rendere la funzione di callback molto più semplice.
Ryan Burn

@rnickb Hai ragione; s->append((char*)contents. nmemb);funziona perfettamente con me ed è più conciso. Inoltre, la firma della funzione ufficiale per il callback ha un char*primo argomento, quindi puoi usarlo e omettere il casting. Infine, s->resize()inizializza effettivamente lo spazio appena allocato. Dato che stai per sovrascriverlo comunque, s->reserve()sarebbe più efficiente.
Jeinzi

Questo mi ha aiutato molto. Puoi anche fare un esempio di come farlo con HTTP POST per favore :-)
Lord Wolfenstein

9

Dalla lettura del manuale qui: http://curl.haxx.se/libcurl/c/curl_easy_setopt.html Penso che tu abbia bisogno di diverse chiamate a CURL_SETOPT, la prima è l'URL che vuoi elaborare, la seconda è qualcosa del tipo:

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, function_ptr);

Dove function_ptr corrisponde a questa firma:

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

Quello che succede qui è che indichi una funzione di callback che libcurl chiamerà quando ha un output da scrivere da qualsiasi trasferimento tu abbia invocato. Puoi fargli scrivere automaticamente in un file o passargli un puntatore a una funzione che gestirà l'output stesso. Usando questa funzione dovresti essere in grado di assemblare le varie stringhe di output in un unico pezzo e quindi usarle nel tuo programma.

Non sono sicuro di quali altre opzioni potresti dover impostare / cos'altro influisce sul modo in cui desideri che la tua app si comporti, quindi dai un'occhiata a quella pagina.


0

Ecco un sapore C ++ della risposta accettata da alex-jasmin

#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;
}
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.