Функция-член класса C ++ и обратный вызов из C API - PullRequest
3 голосов
/ 12 февраля 2012

Я пытаюсь узнать, как вызвать эту функцию write_data(…) из функции funmain() в классе, как показано в коде ниже. (Я знаю, что эта программа работает, если я просто перечислю эти две функции, не помещая ее в класс).

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_data) строка выдает ошибку и не позволяет мне вызывать функцию write_data (…). Можете ли вы исправить мой код и скажите, как мне этого добиться. Любая помощь будет принята с благодарностью. Спасибо.

error C3867: 'go_website::write_data': function call missing argument list; use '&go_website::write_data' to create a pointer to member

//Microsoft Visual Studio 10 in C++
#define CURL_STATICLIB
#include <stdio.h>
#include <curl/curl.h>
#include <curl/types.h>
#include <iostream>
#include <curl/easy.h>
#include <string>
using namespace std;

extern "C" typedef size_t curl_write_callback(void *ptr, size_t size, size_t nmemb, FILE *stream);
class go_website
{
public:
static curl_write_callback write_data;

void funmain()
{
    CURL *curl;
    FILE *fp;
    CURLcode res;
    char *url = "http://www.shorturl.com/";
    char outfilename[FILENAME_MAX] = "C:\\bbb.txt";
    curl = curl_easy_init();
    if (curl) {
        fp = fopen(outfilename,"wb");
        curl_easy_setopt(curl, CURLOPT_URL, url);
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &write_data); 
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, fp);
        res = curl_easy_perform(curl);
        curl_easy_cleanup(curl);
        fclose(fp);
    }
}};

extern "C" size_t write_data(void *ptr, size_t size, size_t nmemb, FILE *stream) 
{
size_t written;
written = fwrite(ptr, size, nmemb, stream);
return written;
}

int main()
{
   go_website a;
   a.funmain();
   return 0;
}

Ответы [ 4 ]

9 голосов
/ 06 сентября 2012

Возможно http://curl.haxx.se/docs/faq.html#Using_C_non_static_functions_f

 // f is the pointer to your object.
 static YourClass::func(void *buffer, size_t sz, size_t n, void *f)
 {
   // Call non-static member function.
   static_cast<YourClass*>(f)->nonStaticFunction();
 }
 // This is how you pass pointer to the static function:
 curl_easy_setopt(hcurl, CURLOPT_WRITEFUNCTION, YourClass:func);
 curl_easy_setopt(hcurl, CURLOPT_WRITEDATA, this);
2 голосов
/ 12 февраля 2012

Эти две строки не будут работать:

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_data);
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, fp);

Второе проще всего исправить: fp это указатель файла, а не функция, вы устанавливаете неправильный атрибут, я думаю, вы хотите CURLOPT_WRITEDATA.

Для функции обратного вызова вам нужен указатель на функцию. Имя обычной функции автоматически распадается на ее адрес, хотя использование оператора адреса (&functionname) более чисто.

Функции-члены класса не исчезают автоматически. Фактически, нестатическая функция-член класса полностью несовместима с обычным указателем на функцию, поскольку нет способа обработать this. К счастью, вам не нужна нестатическая функция-член, поскольку внутри обратного вызова не используются нестатические члены.

Сделайте функцию обратного вызова static и extern "C":

extern "C" typedef size_t curl_write_callback(void *ptr, size_t size, size_t nmemb, FILE *stream);

class go_website
{
public:
    static curl_write_callback write_data;

    // ...
};

extern "C" size_t go_website::write_data(void *ptr, size_t size, size_t nmemb, FILE *stream)
{
   size_t written;
   written = fwrite(ptr, size, nmemb, stream);
   return written;
}

, а затем взять его адрес, используя адрес оператора и полное имя функции:

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &go_website::write_data);

Эта часть была даже объяснена в сообщении об ошибке. Но он не говорил вам, что вам нужно static или extern "C", это проблема списков переменных аргументов, они не безопасны для типов.

После прочтения Стандарта (раздел 7.5 [dcl.link], и особенно параграф 4 и его примеры) это не разрешено. Функция-член по-прежнему имеет связь с языком C ++ как для своего имени (не важно), так и для его типа (это прерыватель сделки).

Вы должны использовать глобальную функцию для обратного вызова:

extern "C" size_t write_data(void *ptr, size_t size, size_t nmemb, FILE *stream)
{
   size_t written;
   written = fwrite(ptr, size, nmemb, stream);
   return written;
}

и затем передайте на него указатель:

curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &write_data);
0 голосов
/ 12 февраля 2012

Проблема в том, что вы хотите, чтобы WRITEFUNCTION была функцией-членом вашего класса.Однако, curl не знает, какой экземпляр класса вызывать.Вам необходимо создать статическую функцию, которую вы передадите в WRITEFUNCTION, а затем передадите указатель this в качестве параметра CURLOPT_WRITEDATA.Затем в вашей статической функции-члене вы можете использовать указатель user_data (который является вашим «this» из WRITE_DATA) в качестве экземпляра класса.

Может быть, этот вопрос поможет: curl WRITEFUNCTION и classes

0 голосов
/ 12 февраля 2012

Я знаю, что эта программа работает, если я просто перечислю эти две функции, не помещая их в класс

Если он работает вне класса, но не внутри, то вам, вероятно, нужно использовать указатель "this".

        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, this->write_data);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...