Wave 文件的前 12 个字节可以这样描述:
TRiff = record ckId : DWORD; {'RIFF'} ckSize : DWORD; {文件大小, 不包括前 8 个字节} fccType : DWORD; {'WAVE'} end;
我们读出文件的前 12 个字节进行判断, 就基本可以确认它是不是 Wave 文件.
uses MMSystem, IOUtils; {这里准备用 IOUtils.TFile.OpenRead 方便地建立文件流} procedure TForm1.FormCreate(Sender: TObject); var riff: record ckId, ckSize, fccType: DWORD; end; {可以同时定义结构并声明结构变量} begin with TFile.OpenRead('C:\WINDOWS\Media\Windows XP 启动.wav') do begin Read(riff, SizeOf(riff)); Free; end; if (riff.ckId = FOURCC_RIFF) and (riff.fccType = mmioStringToFOURCC('WAVE',0)) then ShowMessageFmt('这是个 Wave 文件, 其大小是 %d 字节', [riff.ckSize + 8]); end;
还是把它写成一个函数吧, 最好也别再引用 MMSystem 单元.
{如果是 Wave 文件则返回文件大小, 不是则返回 0} function IsWave(FilePath: string): Integer; function mmioFOURCC(Chr0,Chr1,Chr2,Chr3: AnsiChar): DWORD; begin Result := DWORD(Chr0) + DWORD(Chr1) shl 8 + DWORD(Chr2) shl 16 + DWORD(Chr3) shl 24; end; var riff: record ckId, ckSize, fccType: DWORD; end; begin Result := 0; with TFileStream.Create(FilePath, fmOpenRead) do begin Read(riff, SizeOf(riff)); Free; end; if (riff.ckId = mmioFOURCC('R', 'I', 'F', 'F')) and (riff.fccType = mmioFOURCC('W', 'A', 'V', 'E')) then Result := riff.ckSize + 8; end;
依次道理, 也可以判断一个 RIFF 文件具体是什么格式.
{返回 RIFF 文件格式的函数, 如果不是 RIFF 文件, 则返回 'noneRIFF'} function GetRiffType(FilePath: string): String; function mmioFOURCC(Chr0,Chr1,Chr2,Chr3: AnsiChar): DWORD; begin Result := DWORD(Chr0) + DWORD(Chr1) shl 8 + DWORD(Chr2) shl 16 + DWORD(Chr3) shl 24; end; var riff: record ckId, ckSize, fccType: DWORD; end; type TChars = array[0..3] of AnsiChar; {用于类型转换} begin Result := 'noneRIFF'; with TFileStream.Create(FilePath, fmOpenRead) do begin Read(riff, SizeOf(riff)); Free; end; if (riff.ckId = mmioFOURCC('R', 'I', 'F', 'F')) then Result := TChars(riff.fccType); end; //测试: begin ShowMessage(GetRiffType('C:\WINDOWS\Media\Windows XP 启动.wav')); {WAVE} ShowMessage(GetRiffType('C:\WINDOWS\clock.avi')); {AVI } ShowMessage(GetRiffType('C:\WINDOWS\notepad.exe')); {noneRIFF} end;
关于 FOURCC_RIFF、mmioFOURCC、mmioStringToFOURCC:
RIFF 格式的文件都是有若干 "块" 来构成的, 每个块都是有 4 个字符开头(不足4个字符用空格补足);
这连续的 4 个字节刚好是一个 32 位整数的大小, 所以常常把它们当作一个整数读出来判断.
通过 MMSystem.mmioStringToFOURCC 就可以获取这样的整数.
从 C/C++ 代码中经常看到: mmioFOURCC; 它并非 winmm.dll 库中的函数, 是在 C/C++ 中定义的宏.
这里用 Delphi 模拟实现了这个函数. 其功能类似 mmioStringToFOURCC.
MMSystem.FOURCC_RIFF 是个常量, 当需要 "RIFF" 对应的整数时直接用就是了. 举例:
uses MMSystem; {自定义的 mmioFOURCC 函数} function mmioFOURCC(Chr0,Chr1,Chr2,Chr3: AnsiChar): DWORD; begin Result := DWORD(Chr0) + DWORD(Chr1) shl 8 + DWORD(Chr2) shl 16 + DWORD(Chr3) shl 24; end; procedure TForm1.FormCreate(Sender: TObject); var f1,f2,f3,f4: FOURCC; {FOURCC = DWORD;} begin f1 := mmioStringToFOURCC('RIFF', 0); f2 := mmioStringToFOURCC('Riff', MMIO_TOUPPER); {第二个参数可以把字符串转大写} f3 := mmioFOURCC('R', 'I', 'F', 'F'); f4 := FOURCC_RIFF; ShowMessageFmt('%d, %d, %d, %d', [f1,f2,f3,f4]); {1179011410, 1179011410, 1179011410, 1179011410} end;