libcurl 是底层网络引擎,需手动管理资源;curl_easy_init() 必须配对 curl_easy_cleanup() 防泄漏;POST 数据超2GB时 CURLOPT_POSTFIELDS 失效;SSL 支持需先确认。
libcurl 不是开箱即用的“应用框架”,它是一个 C 语言编写的底层网络传输引擎,直接调用 libcurl 需要手动管理句柄、回调、错误和内存——没封装就别想写得快。
这是最常被忽略的资源泄漏源头。每次 curl_easy_init() 返回一个 CURL* 句柄,它内部持有 DNS 缓存、SSL 上下文、连接池等资源。不调用 curl_easy_cleanup() 就退出,会导致:
Too many open files
即使只发一次请求,也要写成:
struct curl_slist *headers = NULL;
CURL *curl = curl_easy_init();
if (curl) {
curl_easy_setopt(curl, CURLOPT_URL, "https://httpbin.org/get");
curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers);
curl_easy_perform(curl);
curl_easy_cleanup(curl); // 必须有
}
curl_slist_f
ree_all(headers); // 同样不能漏
CURLOPT_POSTFIELDS 接收 void* 和 size_t,但某些旧版 libcurl(long 截断长度,导致超大 body 被截断或触发断言失败。
安全做法是改用 CURLOPT_READFUNCTION 流式上传:
size_t read_callback(void *ptr, size_t size, size_t nmemb, void *userp) {
FILE *fp = (FILE*)userp;
size_t len = fread(ptr, size, nmemb, fp);
if (len == 0 && feof(fp)) return 0;
return len;
}
// 使用时:
FILE *fp = fopen("bigfile.bin", "rb");
curl_easy_setopt(curl, CURLOPT_UPLOAD, 1L);
curl_easy_setopt(curl, CURLOPT_READFUNCTION, read_callback);
curl_easy_setopt(curl, CURLOPT_READDATA, fp);
curl_easy_setopt(curl, CURLOPT_INFILESIZE_LARGE, (curl_off_t)file_size);
CURLOPT_INFILESIZE_LARGE 必须设为 curl_off_t 类型,不能用 CURLOPT_INFILESIZE
fread 返回值需原样返回,libcurl 不做二次校验fp 生命周期必须覆盖整个传输过程libcurl 默认启用 alarm() 和 siglongjmp() 实现超时控制,在多线程程序中(尤其是使用 glibc 的 Linux),这会干扰主线程信号处理,引发随机 crash 或 hang。
正确做法是在每个 CURL* 句柄上显式禁用:
curl_easy_setopt(curl, CURLOPT_NOSIGNAL, 1L);
curl_multi_* 接口,还需确保所有线程共享的 CURLM* 句柄也满足此要求(multi 接口默认已规避 signal)siglongjmp 可能破坏 Objective-C ARC 栈帧,导致野指针关掉证书验证(CURLOPT_SSL_VERIFYPEER, 0L)只是掩盖问题,且会让中间人攻击畅通无阻。真实场景应:
CURLOPT_CAINFO 指向系统可信根证书路径(如 Linux 的 /etc/ssl/certs/ca-bundle.crt)CURLOPT_CAPATH 指向目录(需先用 c_rehash 建索引)CURLOPT_CAINFO
调试阶段可加 CURLOPT_VERBOSE, 1L 查看握手细节,常见失败原因包括:
跨平台打包时,别硬编码证书路径;用 curl_version_info(CURLVERSION_NOW)->features & CURL_VERSION_SSL 先确认 SSL 支持可用,再决定是否加载证书。