標籤:資料存放區 size_t hunk stream https fail test manage 環境
libcurl庫中的參數CURLOPT_WRITEFUNCTION所設定的回呼函數應該是這種:
size_t fun_cb( char *ptr, size_t size, size_t nmemb, void *userdata)
這個回呼函數被調用的時機是有響應資料到達,這些資料由ptr指向,大小是size*nmemb.到這裡為止還是文檔上的說法.從socket的角度考慮,響應資料自然不一定會是以0結尾的字串,而應當被覺得是流資料.僅僅要服務端沒有關閉串連,僅僅要服務端還在發送響應資料,這個函數就會被調用,而被調用的次數不一定僅僅是一次,或許會是很多次,每一次被調用所接收到的資料大小是size*nmemb,這些文檔上似乎沒有提到,由於這是理當如此的事情.所以在處理響應資料的時候不能想當然,必需要考慮到這個函數會被多次調用.
這個回呼函數被調用的時機是有響應資料到達,這些資料由ptr指向,大小是size*nmemb.到這裡為止還是文檔上的說法.從socket的角度考慮,響應資料自然不一定會是以0結尾的字串,而應當被覺得是流資料.僅僅要服務端沒有關閉串連,僅僅要服務端還在發送響應資料,這個函數就會被調用,而被調用的次數不一定僅僅是一次,或許會是很多次,每一次被調用所接收到的資料大小是size*nmemb,這些文檔上似乎沒有提到,由於這是理當如此的事情.所以在處理響應資料的時候不能想當然,必需要考慮到這個函數會被多次調用.
再說說userdata,這是一個FILE *的指標,這個參數跟CURLOPT_WRITEDATA相關,假設已經自己寫了回呼函數,而不是用預設的回呼函數把接收到的資料寫到用CURLOPT_WRITEDATA所設定的userdata所指向的檔案其中去,那麼就能夠把這個指標設為NULL.
範例一:
- char *res_buf = NULL;
- int shift;
- size_t copy_data(void *ptr, size_t size, size_t nmemb, void *stream)
- {
- int res_size;
- res_size = size * nmemb;
- res_buf = realloc(res_buf, shift+res_size + 1);
- memcpy(res_buf + shift, ptr, res_size);
- shift += res_size;
- return size * nmemb;
- }
對於以下的這樣的情況,我不知道是什麼意思:
範例二:
CURL* curl;
CURLcode res;
char buffer[10] ={0};
curl = curl_easy_init();//curl初始化
std::string _version;
if (curl)
{
curl_easy_setopt(curl, CURLOPT_URL, "https://raw.github.com/minggo/AssetsManagerTest/master/version"); curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);//設為不驗證認證
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, getVersionCode);//設定處理資料的函數
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &_version);//資料存放區的對象指標
res = curl_easy_perform(curl);//curl連結
curl_easy_cleanup(curl);//清除curl
}
我想問一下,CURLOPT_WRITEDATA指定的不應該是個檔案指標麼。為什麼是個string型的?
對於範例一涉及到全域變數,因此在多線程環境中是不適合的。以下的範例三是不涉及全域變數的。
範例三:
#include <stdio.h>#include <stdlib.h>#include <string.h> #include <curl/curl.h> struct MemoryStruct { char *memory; size_t size;}; static size_tWriteMemoryCallback(void *contents, size_t size, size_t nmemb, void *userp){ size_t realsize = size * nmemb; struct MemoryStruct *mem = (struct MemoryStruct *)userp; mem->memory = realloc(mem->memory, mem->size + realsize + 1); if(mem->memory == NULL) { /* out of memory! */ printf("not enough memory (realloc returned NULL)\n"); return 0; } memcpy(&(mem->memory[mem->size]), contents, realsize); mem->size += realsize; mem->memory[mem->size] = 0; return realsize;} int main(void){ CURL *curl_handle; CURLcode res; struct MemoryStruct chunk; chunk.memory = malloc(1); chunk.size = 0; curl_global_init(CURL_GLOBAL_ALL); curl_handle = curl_easy_init(); curl_easy_setopt(curl_handle, CURLOPT_URL, "http://www.example.com/"); curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, WriteMemoryCallback); curl_easy_setopt(curl_handle, CURLOPT_WRITEDATA, (void *)&chunk); curl_easy_setopt(curl_handle, CURLOPT_USERAGENT, "libcurl-agent/1.0"); res = curl_easy_perform(curl_handle); if(res != CURLE_OK) { fprintf(stderr, "curl_easy_perform() failed: %s\n", curl_easy_strerror(res)); } else { printf("%lu bytes retrieved\n", (long)chunk.size); } curl_easy_cleanup(curl_handle); free(chunk.memory); curl_global_cleanup(); return 0;}
http get請求擷取server返回的應答資料