1

我正在尝试编写一个 .m 文件以从音轨中提取能量特征,但我似乎在其实现中遇到了麻烦:

% Formula for calculating RMS

[f, fs, nb] = wavread('Three.wav');

frameWidth=441; %10ms
numSamples=length(x);
numFrames=(numSamples/1);
energy(frame)=0;

for frame=1:numFrames,
    startSample=(frame-1)*frameWidth+1;
    endSample=startSample+frameWidth-1;
% Calculate frame energy
    for i=startSample:endSample
        energy(frame)=energy(frame)+x(i)^2;
    end
end

我在 MATLAB 中运行该文件并收到以下错误:

???试图访问 x(2);索引超出范围,因为 numel(x)=1。==> myrms 在 12 能量(帧)=能量(帧)+x(i)^2 处的错误;

任何帮助将非常感激。

4

2 回答 2

2

您应该使用f而不是x,因为f是从您的 .wav 文件加载的实际信号。该变量x可能只是您工作区中的其他一些标量,这就是您收到所见错误的原因。

应该对您的代码进行一些其他更正/改进。首先,正如Paul R 指出的那样,您需要更正您的计算方式numFrames。其次,energy应该初始化为一个零向量。第三,您可以将内部 for 循环简化为单行向量化操作。

以下是我将如何重写您的代码(编辑:根据评论,我更新了代码以保存在循环中计算的一些额外变量):

[y, fs, nb] = wavread('Three.wav');  %# Load the signal into variable y

frameWidth = 441;                          %# 10 msec
numSamples = length(y);                    %# Number of samples in y
numFrames = floor(numSamples/frameWidth);  %# Number of full frames in y
energy = zeros(1,numFrames);               %# Initialize energy
startSample = zeros(1,numFrames);          %# Initialize start indices of frame
endSample = zeros(1,numFrames);            %# Initialize end indices of frame

for frame = 1:numFrames                              %# Loop over frames
  startSample(frame) = (frame-1)*frameWidth+1;       %# Starting index of frame
  endSample(frame) = frame*frameWidth;               %# Ending index of frame
  frameIndex = startSample(frame):endSample(frame);  %# Indices of frame samples
  energy(frame) = sum(y(frameIndex).^2);             %# Calculate frame energy
end
于 2010-08-04T14:16:22.950 回答
0

这行不应该:

numFrames=(numSamples/1);

是这样的:

numFrames=(numSamples / frameWidth);

或者可能:

numFrames=((numSamples + frameWidth - 1) / frameWidth);

?

于 2010-08-04T14:16:26.890 回答