graphics - 绘制离散时间信号显示幅度调制
问题描述
我正在尝试使用canvas
元素渲染一个简单的离散时间信号。但是,这种表示似乎是不准确的。正如您在代码片段中看到的那样,在频率达到某个阈值后,信号似乎被调幅。即使它远低于奈奎斯特极限<50Hz
(假设本例中的采样率为100Hz
)。对于非常低的频率,5Hz
它看起来非常好。
我将如何正确渲染它?它是否适用于更复杂的信号(例如,歌曲的波形)?
window.addEventListener('load', () => {
const canvas = document.querySelector('canvas');
const frequencyElem = document.querySelector('#frequency');
const ctx = canvas.getContext('2d');
const renderFn = t => {
const signal = new Array(100);
const sineOfT = Math.sin(t / 1000 / 8 * Math.PI * 2) * 0.5 + 0.5;
const frequency = sineOfT * 20 + 3;
for (let i = 0; i < signal.length; i++) {
signal[i] = Math.sin(i / signal.length * Math.PI * 2 * frequency);
}
frequencyElem.innerText = `${frequency.toFixed(3)}Hz`
render(ctx, signal);
requestAnimationFrame(renderFn);
};
requestAnimationFrame(renderFn);
});
function render(ctx, signal) {
const w = ctx.canvas.width;
const h = ctx.canvas.height;
ctx.clearRect(0, 0, w, h);
ctx.strokeStyle = 'red';
ctx.beginPath();
signal.forEach((value, i) => {
const x = i / (signal.length - 1) * w;
const y = h - (value + 1) / 2 * h;
if (i === 0) {
ctx.moveTo(x, y);
} else {
ctx.lineTo(x, y);
}
});
ctx.stroke();
}
@media (prefers-color-scheme: dark) {
body {
background-color: #333;
color: #f6f6f6;
}
}
<canvas></canvas>
<br/>
Frequency: <span id="frequency"></span>
解决方案
在我看来是对的。在较高频率下,当峰值落在两个样本之间时,采样点可能比峰值低很多。
如果信号仅具有 < Nyquist 的频率,则可以从其样本中重建信号。这并不意味着样本看起来像信号。
只要您的信号被过采样 2 倍或更多(或更多),您就可以通过在采样点之间使用三次插值来非常准确地绘制它。例如,请参见此处的 Catmull-Rom 插值:https ://en.wikipedia.org/wiki/Cubic_Hermite_spline
您可以使用bezierCurveTo
HTML Canvas 中的方法来绘制这些插值曲线。如果您需要使用线条,那么您应该找到样本之间出现的任何最大或最小点,并将其包含在您的路径中。
我已经编辑了您的代码段以使用bezierCurveTo
下面的 Catmull-Rom 插值方法:
window.addEventListener('load', () => {
const canvas = document.querySelector('canvas');
const frequencyElem = document.querySelector('#frequency');
const ctx = canvas.getContext('2d');
const renderFn = t => {
const signal = new Array(100);
const sineOfT = Math.sin(t / 1000 / 8 * Math.PI * 2) * 0.5 + 0.5;
const frequency = sineOfT * 20 + 3;
for (let i = 0; i < signal.length; i++) {
signal[i] = Math.sin(i / signal.length * Math.PI * 2 * frequency);
}
frequencyElem.innerText = `${frequency.toFixed(3)}Hz`
render(ctx, signal);
requestAnimationFrame(renderFn);
};
requestAnimationFrame(renderFn);
});
function render(ctx, signal) {
const w = ctx.canvas.width;
const h = ctx.canvas.height;
ctx.clearRect(0, 0, w, h);
ctx.strokeStyle = 'red';
ctx.beginPath();
const dx = w/(signal.length - 1);
const dy = -(h-2)/2.0;
const c = 1.0/2.0;
for (let i=0; i < signal.length-1; ++i) {
const x0 = i * dx;
const y0 = h*0.5 + signal[i]*dy;
const x3 = x0 + dx;
const y3 = h*0.5 + signal[i+1]*dy;
let x1,y1,x2,y2;
if (i>0) {
x1 = x0 + dx*c;
y1 = y0 + (signal[i+1] - signal[i-1])*dy*c/2;
} else {
x1 = x0;
y1 = y0;
ctx.moveTo(x0, y0);
}
if (i < signal.length-2) {
x2 = x3 - dx*c;
y2 = y3 - (signal[i+2] - signal[i])*dy*c/2;
} else {
x2 = x3;
y2 = y3;
}
ctx.bezierCurveTo(x1,y1,x2,y2,x3,y3);
}
ctx.stroke();
}
@media (prefers-color-scheme: dark) {
body {
background-color: #333;
color: #f6f6f6;
}
}
<canvas></canvas>
<br/>
Frequency: <span id="frequency"></span>
推荐阅读
- go - 如何编组 XML
- c# - 在excel中以特定格式显示数字
- java - 在 Java 中如何等待所有异步调用完成?
- python - 如何使用 loc 解决 python 中的复制问题?
- asp.net-mvc - 从 ASP.net MVC 中的 IP 地址获取国家/地区的安全方法
- javascript - IE 获取 URL 参数失败
- c++ - 为什么 C/C++ 按位 XOR 运算符关心符号?
- c# - 在 Visual Studio 的包管理器控制台上执行“更新数据库”时出现错误:找不到内容根文件夹
- c# - 将两个csv文件导入mysql数据库
- vba - 使用 vba 在 Outlook 2016 中保存来自非默认邮箱的收入消息