C libcurl รับเอาต์พุตเป็นสตริง


98

ฉันต้องการเก็บผลลัพธ์ของฟังก์ชัน 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

1
คำถามเดียวกัน แต่สำหรับ C ++ แทนที่จะเป็น c ไปที่นี่: บันทึกผลลัพธ์เนื้อหา cURL ลงในสตริงใน C ++
Trevor Boyd Smith

คำตอบ:


117

คุณสามารถตั้งค่าฟังก์ชันการโทรกลับเพื่อรับชิ้นข้อมูลขาเข้าโดยใช้ 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

2
สำหรับ C ++ std::stringไปที่นี่
Trevor Boyd Smith

35

คำตอบต่อไปนี้เป็นวิธี 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 ได้หรือไม่ :-)
Lord 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 จะเรียกใช้เมื่อมีเอาต์พุตบางอย่างที่จะเขียนจากการถ่ายโอนใด ๆ ที่คุณเรียกใช้ คุณสามารถกำหนดให้มันเขียนไปยังไฟล์โดยอัตโนมัติหรือส่งตัวชี้ไปยังฟังก์ชันที่จะจัดการกับผลลัพธ์เอง การใช้ฟังก์ชั่นนี้คุณควรจะสามารถรวบรวมสตริงเอาต์พุตต่างๆให้เป็นชิ้นเดียวแล้วใช้ในโปรแกรมของคุณ

ฉันไม่แน่ใจว่าตัวเลือกอื่น ๆ ที่คุณอาจต้องตั้งค่า / มีอะไรอีกบ้างที่ส่งผลต่อวิธีที่คุณต้องการให้แอปของคุณทำงานดังนั้นโปรดดูให้ดีในหน้านั้น


0

นี่คือรสชาติ C ++ ของคำตอบที่ยอมรับจาก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;
}

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.