Wave 文件(15): 合并与剪裁 wav 文件
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type TForm1 = class(TForm) Button1: TButton; Button2: TButton; procedure Button1Click(Sender: TObject); procedure Button2Click(Sender: TObject); end; var Form1: TForm1; implementation {$R *.dfm} uses MMSystem; //从指定 wav 文件中获取格式信息和波形数据的函数 function GetWaveFmtData(const path: string; var fmt: TWaveFormatEx; var buf: TBytes): Boolean; var hFile: HMMIO; ckiRIFF,ckiFmt,ckiData: TMMCKInfo; begin Result := False; hFile := mmioOpen(PChar(path), nil, MMIO_READ); if hFile = 0 then Exit; ZeroMemory(@ckiRIFF, SizeOf(TMMCKInfo)); ZeroMemory(@ckiFmt, SizeOf(TMMCKInfo)); ZeroMemory(@ckiData, SizeOf(TMMCKInfo)); ckiRIFF.fccType := mmioStringToFOURCC('WAVE', 0); ckiFmt.ckid := mmioStringToFOURCC('fmt', 0); ckiData.ckid := mmioStringToFOURCC('data', 0); ZeroMemory(@fmt, SizeOf(TWaveFormatEx)); mmioDescend(hFile, @ckiRIFF, nil, MMIO_FINDRIFF); if (ckiRIFF.ckid = FOURCC_RIFF) and (ckiRIFF.fccType = mmioStringToFOURCC('WAVE',0)) and (mmioDescend(hFile, @ckiFmt, @ckiRIFF, MMIO_FINDCHUNK) = MMSYSERR_NOERROR) and (mmioRead(hFile, @fmt, ckiFmt.cksize) = ckiFmt.cksize) and (mmioAscend(hFile, @ckiFmt, 0) = MMSYSERR_NOERROR) and (mmioDescend(hFile, @ckiData, @ckiRIFF, MMIO_FINDCHUNK) = MMSYSERR_NOERROR) then begin SetLength(buf, ckiData.cksize); Result := (mmioRead(hFile, PAnsiChar(buf), ckiData.cksize) = ckiData.cksize); end; mmioClose(hFile, 0); end; //根据格式信息和波形数据建立 wav 文件的函数 function CreateWave(const path: string; const fmt: TWaveFormatEx; const buf: TBytes): Boolean; var h: HMMIO; ckiRiff, ckiFmt, ckiData: TMMCKInfo; begin ZeroMemory(@ckiRiff, SizeOf(TMMCKInfo)); ckiRiff.cksize := 44 - 8 + Length(buf); ckiRiff.fccType := mmioStringToFOURCC('WAVE', 0); ZeroMemory(@ckiFmt, SizeOf(TMMCKInfo)); ckiFmt.ckid := mmioStringToFOURCC('fmt', 0); ZeroMemory(@ckiData, SizeOf(TMMCKInfo)); ckiData.ckid := mmioStringToFOURCC('data', 0); ckiData.cksize := Length(buf); h := mmioOpen(PChar(path), nil, MMIO_CREATE or MMIO_WRITE); if (h <> 0) and (mmioCreateChunk(h, @ckiRiff, MMIO_CREATERIFF) = MMSYSERR_NOERROR) and (mmioCreateChunk(h, @ckiFmt, 0) = MMSYSERR_NOERROR) and (mmioWrite(h, PAnsiChar(@fmt), SizeOf(TPCMWaveFormat)) = SizeOf(TPCMWaveFormat)) and (mmioAscend(h, @ckiFmt, 0) = MMSYSERR_NOERROR) and (mmioCreateChunk(h, @ckiData, 0) = MMSYSERR_NOERROR) then Result := (mmioWrite(h, PAnsiChar(buf), Length(buf)) = Length(buf)); mmioClose(h, 0); end; //截取 wav 文件, 本例截留了文件的 1/4 procedure TForm1.Button1Click(Sender: TObject); const pathSource = 'C:\WINDOWS\Media\Windows XP 启动.wav'; pathDest = 'C:\Temp\New1.wav'; var fmt: TWaveFormatEx; buf: TBytes; begin GetWaveFmtData(pathSource, fmt, buf); SetLength(buf, Length(buf) div 4); CreateWave(pathDest, fmt, buf); end; //合并 wav 文件 procedure TForm1.Button2Click(Sender: TObject); const path1 = 'C:\WINDOWS\Media\Windows XP 启动.wav'; path2 = 'C:\WINDOWS\Media\Windows XP 关机.wav'; pathDest = 'C:\Temp\New2.wav'; var fmt1,fmt2: TWaveFormatEx; buf1,buf2: TBytes; oldLen: Integer; begin GetWaveFmtData(path1, fmt1, buf1); GetWaveFmtData(path2, fmt2, buf2); if CompareMem(@fmt1, @fmt2, SizeOf(TWaveFormatEx)) then begin oldLen := Length(buf1); SetLength(buf1, Length(buf1) + Length(buf2)); CopyMemory(@buf1[oldLen], Pointer(buf2), Length(buf2)); CreateWave(pathDest, fmt1, buf1); end else ShowMessage('文件格式不一致, 没有执行合并!'); end; end.
欢迎沟通及共同进步
QQ:1441299
Email:1441299@qq.com
越折腾越快乐!
分类:
vb
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 周边上新:园子的第一款马克杯温暖上架
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?
· 使用C#创建一个MCP客户端