使用百度的长文本转语音API时无法下载.MP3文件

今天是学生们交作业的时候,结果是我最忙碌的一天,各种改bug。

有个学生来问:

我在百度提供的API代码(长文本转语音)的基础上添加了下载生成的.MP3文件的代码,运行之后成功建成了.MP3文件,但是文件的内容确实以下的报错信息,根据这一信息采用了新的accesstoken之后依旧无法解决。

看了输出,发现获取链接的部分已经完成,是后面下载这一链接里MP3文件的代码中存在的问题,但是经过查询都没有指出这一部分代码的问题

从这个错误信息看,似乎是下载MP3文件时,百度的服务器认为链接非法,所以拒绝了。但是如果单独用curl或者wget去下载这个MP3的链接,是可以得到正确的文件的。看来这个问题是在程序里面隐藏的。

要来了学生的代码,大致是这样的:

cpp 复制代码
int main() {
    std::cout << "Current working directory: " << getcwd(NULL, 0) << std::endl;
    CURL *curl;
    FILE *fp;

    CURLcode res;
    std::string result;
    std::string downloadUrl;
    std::string outputFilename = "output.mp3";
    curl_global_init(CURL_GLOBAL_ALL);
    curl = curl_easy_init();
    if(curl) {
        // 发送 POST 请求获取语音合成任务结果
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L); // 禁用证书验证
        curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L); // 禁用证书验证
        curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "POST");
        // URL和access_token
        curl_easy_setopt(curl, CURLOPT_URL, "https://aip.baidubce.com/rpc/2.0/tts/v1/query?access_token=xxxxxxxx");
        curl_easy_setopt(curl, CURLOPT_POSTFIELDS,"{\"task_ids\":[\"yyyyyyy\"]}");
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, &result);
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, onWriteData);
        res = curl_easy_perform(curl);
        if(res != CURLE_OK) {
            std::cerr << "curl_easy_perform() failed: " << curl_easy_strerror(res) << std::endl;
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }

        // 解析 JSON 响应,获取 MP3 文件的下载链接
        cJSON *json = cJSON_Parse(result.c_str());
        if (json == NULL) {
            std::cerr << "Error parsing JSON." << std::endl;
            // 清理工作
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }

......

        cJSON *task_result = cJSON_GetObjectItem(task_info, "task_result");
        if (task_result != NULL) {
            // "speech_url",字段名
            cJSON *download_url_json = cJSON_GetObjectItem(task_result, "speech_url");
	    if (download_url_json != NULL && download_url_json->type == cJSON_String) {
                downloadUrl = download_url_json->valuestring; // 获取下载链接
                //fprintf(stderr, "speech-url: %s\n", downloadUrl.c_str());
	    } else {
                std::cerr << "No download URL found in task result." << std::endl;
                // 清理
                cJSON_Delete(json);
                curl_easy_cleanup(curl);
                curl_global_cleanup();
                return 1;
            }
        } else {
            std::cerr << "No task result found." << std::endl;
            // 清理
            cJSON_Delete(json);
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }
        // 下载 MP3 文件
	fp = fopen(outputFilename.c_str(), "wb");
        if (fp==nullptr) {
            std::cerr << "Failed to open file for writing." << std::endl;
            // 清理
            cJSON_Delete(json);
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }
        curl_easy_setopt(curl, CURLOPT_URL, downloadUrl.c_str()); // 下载链接
        fprintf(stderr, "speech-url: %s\n", downloadUrl.c_str()); // 打印查看
        curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, WriteCallback);
        curl_easy_setopt(curl, CURLOPT_WRITEDATA, fp);
	res = curl_easy_perform(curl); // 下载
        if (res != CURLE_OK) {
            std::cerr << "curl_easy_perform() failed: " << curl_easy_strerror(res) << std::endl;
            fclose(fp);
            // 清理
            cJSON_Delete(json);
            curl_easy_cleanup(curl);
            curl_global_cleanup();
            return 1;
        }

从这段代码可以看出,curl变量前面被用来做POST请求,后面又被用来做GET请求,第一次操作的状态影响了第二次,导致第二次发给服务器的数据错了。让学生改成下面这样就对了:

当然更好的办法是用两个curl变量来完成不同的任务。

一定要记住,不要重复使用一个变量完成两个不同的任务!

相关推荐
xixixi777774 分钟前
Hugging Face 开源模型供应链安全平台、AI 数据隐私保护法草案发布:AI 安全合规进入技术 + 法律双驱动阶段
大数据·人工智能·安全·ai·开源·大模型·智能体
真实的菜5 分钟前
Deepseek V4 Flash 深度评测:参数解析与实战边界
人工智能
扫地僧9855 分钟前
基于改进版YOLOv11的海洋垃圾检测系统设计与实现
人工智能·深度学习·yolo
ZHW_AI课题组5 分钟前
基于XGBoost的鸢尾花花瓣长度回归预测
人工智能·数据挖掘·回归
moMo6 分钟前
藏好Key的小妙招,从搭建node.js+大模型项目开始
人工智能·node.js
前端摸鱼匠6 分钟前
YOLOv11 深入 Ultralytics 框架的源码目录,解析 ultralytics/cfg/models/11/ 下的模型配置文件,以及 ultralytics/nn/modules/下的模块
人工智能·yolo·目标检测·计算机视觉·目标跟踪
KaMeidebaby6 分钟前
卡梅德生物技术快报|组蛋白乙酰化修饰调控动脉粥样硬化的分子机制及中药表观干预研究
网络·人工智能·网络协议·tcp/ip·算法
SEO_juper6 分钟前
搜索进入 Agentic 智能体时代,内容要能 “被 AI 直接用”
人工智能·ai·seo·跨境电商·geo·谷歌优化·2026
装不满的克莱因瓶7 分钟前
机器学习和数据科学的基石:NumPy详解与实战技巧
人工智能·线性代数·机器学习·ai·矩阵·numpy
好好风格8 分钟前
微软这个 14 万星工具,把 PDF、PPT、Excel 都变成大模型爱读的 Markdown
人工智能·python·开源