C libcurlは文字列に出力を取得します


93

このカール関数の結果を変数に格納したいのですが、どうすればよいですか?

#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

1
CURLcode res;未使用
fnc12 2016

同じ質問ですが、CではなくC ++の場合は、ここに移動します。cURLコンテンツの結果をC ++の文字列に保存します
Trevor Boyd Smith

回答:


112

を使用して着信データチャンクを受信するためのコールバック関数を設定できます。 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

次の答えは、std::stringヌルで終わる文字列の代わりに、C ++でそれを行う方法です。それでもコールバック関数は使用されますが(回避策はありません)、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 2018

@rnickbその通りです。s->append((char*)contents. nmemb);私と完璧に連携し、より簡潔です。また、コールバックの公式関数シグネチャにはchar*最初の引数としてaがあるため、それを使用してキャストを省略できます。そして最後に、s->resize()新しく割り当てられたスペースを実際に初期化します。とにかく上書きs->reserve()しようとしているので、より効率的です。
Jeinzi

これは私を大いに助けました。HTTP POSTでそれを行う方法の例も教えてください:-)
ヴォルフェンシュタイン卿

9

ここのマニュアルを読むことから:http : //curl.haxx.se/libcurl/c/curl_easy_setopt.html CURL_SETOPTへのいくつかの呼び出しが必要だと思います。最初は処理したいURLで、2番目は次のようなものです。

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, function_ptr);

function_ptrがこのシグネチャと一致する場合:

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

ここで何が起こるかは、あなたが呼び出した転送から書き込む出力があるときにlibcurlが呼び出すコールバック関数を示すことです。自動的にファイルに書き込むか、出力自体を処理する関数へのポインタを渡すことができます。この関数を使用すると、さまざまな出力文字列を1つにまとめて、プログラムで使用できます。

他にどのオプションを設定する必要があるか、他にどのようにアプリの動作に影響するかはわかりません。そのページをよく確認してください。


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

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.