
在使用MediaRecorder进行实时音频录制并分块上传至服务器时,常见的错误是生成的音频文件无法播放。本文将深入探讨导致此问题的原因,即MediaRecorder的`mimeType`配置不当以及服务器端文件写入方式不正确。我们将提供一套完整的解决方案,包括客户端JavaScript的MediaRecorder初始化配置、数据处理以及服务器端PHP的正确文件追加逻辑,确保实时录制的音频能够成功保存并播放。
当开发者尝试通过JavaScript的MediaRecorder API实时捕获麦克风音频,将其分块编码为Base64,并通过POST请求发送到服务器,再由服务器解码并写入文件时,经常会遇到保存的.ogg或其它格式音频文件损坏、无法播放的问题。
核心原因通常有两个:
要解决MediaRecorder的MIME类型配置问题,我们需要在实例化MediaRecorder对象时,通过第二个参数传入一个配置对象,其中包含mimeType属性。
以下是修正后的JavaScript客户端代码示例:
var mediaRecorder = null;
let chunks = [];
if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
console.log('getUserMedia supported.');
navigator.mediaDevices.getUserMedia(
{
audio: true
})
.then(function(stream) {
// 关键:在MediaRecorder构造函数中指定mimeType
const mrOptions = { mimeType: 'audio/ogg; codecs=opus' };
mediaRecorder = new MediaRecorder(stream, mrOptions);
// 每2秒触发一次ondataavailable事件,收集数据块
mediaRecorder.start(2000);
mediaRecorder.ondataavailable = function(e) {
// 确保e.data中有数据
if (e.data.size > 0) {
chunks.push(e.data);
// 使用MediaRecorder实例的mimeType来创建Blob
const blob = new Blob(chunks, { type : mediaRecorder.mimeType });
chunks = []; // 清空chunks,准备接收下一个数据块
var reader = new FileReader();
reader.readAsDataURL(blob);
reader.onloadend = function() {
// 提取Base64编码的数据
var data = reader.result.split(";base64,")[1];
// 发送数据到服务器
requestp2("a.php", "data=" + encodeURIComponent(data));
}
}
};
mediaRecorder.onstop = function() {
console.log("录制停止");
// 可以在这里处理最后剩余的chunks,如果需要
};
})
.catch(function(err) {
console.log('获取麦克风权限失败: ' + err);
}
);
} else {
console.log('当前浏览器不支持getUserMedia!');
}
/**
* 封装的POST请求函数
* @param {string} path 请求路径
* @param {string} data 要发送的数据
*/
function requestp2(path, data) {
var http = new XMLHttpRequest();
http.open('POST', path, true);
http.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');
http.onreadystatechange = function() {
if (http.readyState === 4 && http.status === 200) {
console.log("数据发送成功", http.responseText);
} else if (http.readyState === 4 && http.status !== 200) {
console.error("数据发送失败", http.status);
}
};
http.send(data);
}代码解释:
在服务器端,PHP脚本需要接收Base64编码的数据,对其进行解码,并将其追加到文件中,而不是每次都覆盖。
以下是修正后的PHP服务器端代码示例:
<?php
// 定义保存音频文件的路径和文件名
$audioFilePath = "r.ogg";
if (isset($_POST["data"])) {
$base64Data = $_POST["data"];
// 解码Base64数据
$decodedData = base64_decode($base64Data);
if ($decodedData === false) {
error_log("Base64解码失败!");
http_response_code(400); // Bad Request
echo "Error: Base64 decoding failed.";
exit;
}
// 关键:使用FILE_APPEND标志将数据追加到文件末尾
// 如果文件不存在,file_put_contents 会尝试创建它
if (file_put_contents($audioFilePath, $decodedData, FILE_APPEND | LOCK_EX) === false) {
error_log("文件写入失败:" . $audioFilePath);
http_response_code(500); // Internal Server Error
echo "Error: Failed to write to file.";
exit;
}
echo "Success: Audio chunk saved.";
exit;
} else {
http_response_code(400); // Bad Request
echo "Error: No data received.";
exit;
}
?>代码解释:
通过正确配置MediaRecorder的mimeType并在服务器端使用FILE_APPEND模式追加数据,可以有效地解决实时分块录制音频文件损坏的问题。这个方案不仅保证了录制音频的完整性和可播放性,也为构建基于Web的实时语音应用奠定了坚实基础。在实际开发中,务必结合错误处理、安全性考量和性能优化,以提供稳定可靠的用户体验。
以上就是MediaRecorder实时音频分块录制与服务器端保存:解决文件损坏问题的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号