libcurl C++ 的内存访问错误
Memory access error with libcurl C++
我正在尝试制作一个非常简单的程序来获取用户请求(控制台)并将此请求发送到 headers (libcurl) 中的网站,然后获取响应 headers。
所以我有2个类(其实我得到了很多类但是对于这里的问题我会简化)
我的 main.cpp 收到请求并将其发送到我的服务。
这是我的服务 Fuzzer.h :
#ifndef FUZZER_H
#define FUZZER_H
#include <string>
using namespace std;
namespace furez
{
class Fuzzer {
private:
string url;
string response;
static size_t writeH_static(void *, size_t, size_t, void *);
public:
Fuzzer(string);
virtual ~Fuzzer();
string sendRequest(string);
size_t writeH(void *, size_t, size_t);
};
}
#endif
这是我的 Fuzzer.cpp:
#include <string>
#include <curl/curl.h>
#include <iostream>
#include "Fuzzer.h"
namespace furez
{
Fuzzer::Fuzzer(std::string u)
{
this->url = u;
}
Fuzzer::~Fuzzer(){}
string Fuzzer::sendRequest(std::string request)
{
CURL *curl;
CURLcode res;
static const char *headerfilename = "head.out";
static const char *bodyfilename = "body.out";
curl = curl_easy_init();
if (curl) {
struct curl_slist *chunk = NULL;
chunk = curl_slist_append(chunk, "Accept:");
chunk = curl_slist_append(chunk, "Another: yes");
chunk = curl_slist_append(chunk, "Host: example.com");
chunk = curl_slist_append(chunk, "X-silly-header;");
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &Fuzzer::writeH);
Fuzzer instance(request);
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &instance);
curl_easy_setopt(curl, CURLOPT_URL, "localhost");
res = curl_easy_perform(curl);
/* Check for errors */
if (res != CURLE_OK) {
fprintf(stderr, "curl_easy_perform() failed: %s\n",
curl_easy_strerror(res));
}
curl_easy_cleanup(curl);
curl_slist_free_all(chunk);
}
return request;
}
size_t Fuzzer::writeH_static(void *ptr, size_t size, size_t nmemb, void *stream)
{
return ((Fuzzer*)stream)->writeH(ptr, size, nmemb);
}
size_t Fuzzer::writeH(void *ptr, size_t size, size_t nmemb)
{
//std::string str = (CHAR *)ptr;
//std::string str2("Server:");
//if (str.find(str2) != string::npos) {
//this->response = str;
//}
size_t realsize = size * nmemb;
return realsize;
}
}
但是当我尝试执行此操作时出现内存错误。
正因如此:
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &Fuzzer::writeH);
我正在为我的项目使用 libcurl。我知道这是一个 C 库,也许这是问题所在?
为了帮助我,我遵循了这些说明:
libcurl callbacks w/c++ class members
还有这个:
curl WRITEFUNCTION and classes
我在 Windows 7 上 Visual Studio 2013.
(PS: 当我把没有 &Fuzzer::writeH 和静态函数的代码放在 main.cpp 里面时,它可以正常工作。)
我认为问题出在这里:
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &Fuzzer::writeH);
正如您正确指出的那样,libcurl 是一个 C 库,但是您向函数 curl_easy_setopt 传递了一个指向成员函数的指针。
现在,函数指针和指向 class 成员的指针是不同的类型,因此,您不能那样做。
我知道的唯一解决方案是使用 trampoline 调用.
- 创建一个普通函数(非您 class 的成员),您将在 curl_easy_setopt()
中将其注册为回调
- 让你的 class 成为蹦床回调好友(这样她就可以在需要时访问私人成员)
- 传递给回调的 userdata 参数应该是指向您的 class
的指针
- 在蹦床回调中,您将指针投回到您的 class 并调用所需的成员函数
举个例子:
extern "C" size_t trampolineCallback( char * i_buffer, size_t i_size, size_t i_nitems, void * i_userdata );
在您的 class 实施中,您将拥有类似的内容:
curl_easy_setopt( m_curlHndlPtr, CURLOPT_HEADERDATA, (void*) this );
curl_easy_setopt( m_curlHndlPtr, CURLOPT_HEADERFUNCTION, trampolineCallback );
您的蹦床实现将如下所示:
size_t trampolineCallback(char *i_buffer, size_t i_size, size_t i_nitems, void *i_userdata)
{
MyObj * object = (MyObj*) i_userdata;
object->myMemberFunc( i_buffer,i_size, i_nitems );
return (i_size * i_nitems);
}
我正在尝试制作一个非常简单的程序来获取用户请求(控制台)并将此请求发送到 headers (libcurl) 中的网站,然后获取响应 headers。
所以我有2个类(其实我得到了很多类但是对于这里的问题我会简化)
我的 main.cpp 收到请求并将其发送到我的服务。
这是我的服务 Fuzzer.h :
#ifndef FUZZER_H
#define FUZZER_H
#include <string>
using namespace std;
namespace furez
{
class Fuzzer {
private:
string url;
string response;
static size_t writeH_static(void *, size_t, size_t, void *);
public:
Fuzzer(string);
virtual ~Fuzzer();
string sendRequest(string);
size_t writeH(void *, size_t, size_t);
};
}
#endif
这是我的 Fuzzer.cpp:
#include <string>
#include <curl/curl.h>
#include <iostream>
#include "Fuzzer.h"
namespace furez
{
Fuzzer::Fuzzer(std::string u)
{
this->url = u;
}
Fuzzer::~Fuzzer(){}
string Fuzzer::sendRequest(std::string request)
{
CURL *curl;
CURLcode res;
static const char *headerfilename = "head.out";
static const char *bodyfilename = "body.out";
curl = curl_easy_init();
if (curl) {
struct curl_slist *chunk = NULL;
chunk = curl_slist_append(chunk, "Accept:");
chunk = curl_slist_append(chunk, "Another: yes");
chunk = curl_slist_append(chunk, "Host: example.com");
chunk = curl_slist_append(chunk, "X-silly-header;");
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &Fuzzer::writeH);
Fuzzer instance(request);
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &instance);
curl_easy_setopt(curl, CURLOPT_URL, "localhost");
res = curl_easy_perform(curl);
/* Check for errors */
if (res != CURLE_OK) {
fprintf(stderr, "curl_easy_perform() failed: %s\n",
curl_easy_strerror(res));
}
curl_easy_cleanup(curl);
curl_slist_free_all(chunk);
}
return request;
}
size_t Fuzzer::writeH_static(void *ptr, size_t size, size_t nmemb, void *stream)
{
return ((Fuzzer*)stream)->writeH(ptr, size, nmemb);
}
size_t Fuzzer::writeH(void *ptr, size_t size, size_t nmemb)
{
//std::string str = (CHAR *)ptr;
//std::string str2("Server:");
//if (str.find(str2) != string::npos) {
//this->response = str;
//}
size_t realsize = size * nmemb;
return realsize;
}
}
但是当我尝试执行此操作时出现内存错误。 正因如此:
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &Fuzzer::writeH);
我正在为我的项目使用 libcurl。我知道这是一个 C 库,也许这是问题所在?
为了帮助我,我遵循了这些说明: libcurl callbacks w/c++ class members
还有这个: curl WRITEFUNCTION and classes 我在 Windows 7 上 Visual Studio 2013.
(PS: 当我把没有 &Fuzzer::writeH 和静态函数的代码放在 main.cpp 里面时,它可以正常工作。)
我认为问题出在这里:
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, &Fuzzer::writeH);
正如您正确指出的那样,libcurl 是一个 C 库,但是您向函数 curl_easy_setopt 传递了一个指向成员函数的指针。 现在,函数指针和指向 class 成员的指针是不同的类型,因此,您不能那样做。 我知道的唯一解决方案是使用 trampoline 调用.
- 创建一个普通函数(非您 class 的成员),您将在 curl_easy_setopt() 中将其注册为回调
- 让你的 class 成为蹦床回调好友(这样她就可以在需要时访问私人成员)
- 传递给回调的 userdata 参数应该是指向您的 class 的指针
- 在蹦床回调中,您将指针投回到您的 class 并调用所需的成员函数
举个例子:
extern "C" size_t trampolineCallback( char * i_buffer, size_t i_size, size_t i_nitems, void * i_userdata );
在您的 class 实施中,您将拥有类似的内容:
curl_easy_setopt( m_curlHndlPtr, CURLOPT_HEADERDATA, (void*) this );
curl_easy_setopt( m_curlHndlPtr, CURLOPT_HEADERFUNCTION, trampolineCallback );
您的蹦床实现将如下所示:
size_t trampolineCallback(char *i_buffer, size_t i_size, size_t i_nitems, void *i_userdata)
{
MyObj * object = (MyObj*) i_userdata;
object->myMemberFunc( i_buffer,i_size, i_nitems );
return (i_size * i_nitems);
}