使用javascript切分录音的blob流并生成5秒的wav文件
在使用react-mic进行录音时,遇到一个需求:需要将录音的blob流切分成多个5秒的wav文件。然而,尝试之后发现只有第一个切分的wav文件能够正常播放,其余文件均提示文件损坏。
在前端实现这个需求时,主要面临两个挑战:一是如何正确切分blob流,二是如何确保每个切分后的片段能够正确生成并播放wav文件。以下是代码示例和解决思路:
import react, { useref, usestate } from 'react'
import { reactmic, reactmicstopevent } from 'react-mic'
import { button } from 'antd'
const audiorecorder = () => {
const [record, setrecord] = usestate(false)
const resref = useref<blob[]>([])
const audiochunksref = useref<blob[]>([])
const intervalref = useref<nodejs.timer | null>(null)
const firstblob = useref<blob | undefined>(undefined)
const createwavheader = (numchannels, samplerate, bytelength) => {
const header = new arraybuffer(44);
const view = new dataview(header);
view.setuint32(0, 1380533830, false); // "riff"
view.setuint32(4, bytelength + 36, false);
view.setuint32(8, 1718449184, false); // "wave"
view.setuint32(12, 1684108385, false); // "fmt "
view.setuint32(16, 16, true); // 16 for pcm
view.setuint16(20, 1, true); // pcm
view.setuint16(22, numchannels, true);
view.setuint32(24, samplerate, true);
view.setuint32(28, samplerate * numchannels * 2, true);
view.setuint16(32, numchannels * 2, true);
view.setuint16(34, 16, true); // 16 bits
view.setuint32(36, 1684108385, false); // "data"
view.setuint32(40, bytelength, true);
return header;
};
const savefile = async () => {
const chunkslist = resref.current;
for (let i = 0; i < chunkslist.length; i++) {
const audiobuffer = new uint8array(await chunkslist[i].arraybuffer());
const header = createwavheader(1, 44100, audiobuffer.length); // 假设单声道和 44100hz
const wavblob = new blob([header, audiobuffer], { type: 'audio/wav' });
const url = url.createobjecturl(wavblob);
const a = document.createelement('a');
a.href = url;
a.download = `recording${i}.wav`;
a.click();
url.revokeobjecturl(url);
}
};
const startrecording = () => {
setrecord(true)
audiochunksref.current = [] // 清空之前的录音数据
// 每5秒分割一次录音
intervalref.current = setinterval(() => {
const curblob = new blob(audiochunksref.current, { type: 'audio/wav' })
const startindex = audiochunksref.current.indexof(firstblob.current as blob)
const blob = curblob.slice(startindex === -1 ? 0 : startindex, -1, 'audio/wav')
firstblob.current = audiochunksref.current.at(-1)
// 处理当前录音数据
console.log('分割当前录音数据:', blob)
resref.current.push(blob)
}, 5000)
}
const stoprecording = () => {
setrecord(false)
intervalref.current && clearinterval(intervalref.current) // 清除定时器
}
const ondata = (recordedblob: blob) => {
audiochunksref.current.push(recordedblob) // 保存录音数据
}
const onstop = (recordedblob: reactmicstopevent) => {
console.log('录音完成:', recordedblob)
}
const savefile1 = () => {
const chunkslist = resref.current
chunkslist.map(async (v, i) => {
const filename = 'aaa.wav'
const file: file = new file([v], filename, { type: 'audio/wav' })
const filesize = file.size
console.log('filesize', filesize)
// 创建下载链接
const url = url.createobjecturl(file)
const a = document.createelement('a')
a.href = url
a.download = `recording${i}.wav` // 设置下载文件的名称
a.click() // 触发下载
// 释放url资源
url.revokeobjecturl(url)
})
}
const savefinalresult = () => {
const filename = 'aaa.wav'
const file: file = new file(audiochunksref.current, filename, { type: 'audio/wav' })
const filesize = file.size
console.log('filesize', filesize)
// 创建下载链接
const url = url.createobjecturl(file)
const a = document.createelement('a')
a.href = url
a.download = `recording${date.now()}.wav` // 设置下载文件的名称
a.click() // 触发下载
// 释放url资源
url.revokeobjecturl(url)
}
return (
<div>
<reactmic record={record} onstop={onstop} ondata={ondata} mimetype="audio/wav" />
<button onclick={startrecording}>开始录音</button>
<button onclick={stoprecording}>停止录音</button>
<button onclick={savefile}>下载</button>
<button onclick={savefinalresult}>下载final</button>
</div>
)
}
export default audiorecorder在尝试切分blob流和生成wav文件的过程中,发现手动添加wav头信息并不能解决问题。其原因在于wav文件的结构比较严格,切分后如果不正确地添加头部信息,文件可能会损坏。
解决这个问题的一个建议是使用ffmpeg的wasm版本,这是一个可以在浏览器中运行的音视频处理库。通过它,你可以轻松地对音频进行切分并生成正确的wav文件格式。可以考虑使用ffmpeg.wasm项目来实现这个功能。
以上就是如何使用javascript将录音的blob流切分成多个5秒的wav文件并确保其正常播放?的详细内容,更多请关注代码网其它相关文章!
发表评论