4

我正在尝试学习如何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

4 回答 4

10

有可能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);
于 2012-09-06T16:55:15.513 回答
2

这两行不起作用:

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

然后使用 address-of 运算符和完全限定的函数名获取其地址:

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

这部分甚至在错误消息中进行了解释。但它没有告诉你需要staticor 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);
于 2012-02-12T05:52:39.240 回答
1

我知道如果我只列出这两个函数而不把它放在一个类中,这个程序就可以工作

如果它在类外工作,但不在类内,那么您可能需要使用“this”指针。

        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, this->write_data);
于 2012-02-12T05:47:00.973 回答
0

问题是您希望 WRITEFUNCTION 成为您的类的成员函数。然而,curl 不知道在哪个类的实例上调用它。您需要创建一个传递给 WRITEFUNCTION 的静态函数,然后将 this 指针作为 CURLOPT_WRITEDATA 参数传递。然后在您的静态成员函数中,您可以使用 user_data 指针(这是来自 WRITE_DATA 的“this”)作为类的实例。

也许这个问题会有所帮助:curl WRITEFUNCTION and classes

于 2012-02-12T05:55:46.223 回答