- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个分析生成的频谱,其中 x 轴表示角频率,y 表示强度。频谱以某个频率值为中心,这通常称为信号的中心频率(图片上的蓝色图表)。
我想对数据执行 IFFT 到时域,用高斯曲线切割它的有用部分,然后 FFT 回到原始域。我的问题是,在 IFFT(FFT(signal)) 之后,中心频率丢失了:我按形状取回频谱,但它始终以 0 为中心(橙色图)。
目前我对此的解决方案非常糟糕:我缓存了原始的 x 轴,并在 FFT 调用时恢复它。这显然有很多缺点,我想改进它。下面我包含了一个演示问题的小演示。我的问题是:这可以以更优雅的方式解决吗?有没有办法在这个过程中不丢失中心频率?
import numpy as np
from scipy.fftpack import fft, ifft, fftshift, fftfreq
from scipy.interpolate import interp1d
import matplotlib.pyplot as plt
C_LIGHT = 299.793
def generate_data(start, stop, center, delay, GD=0, resolution=0.1):
window = 8 * np.log(2) / 50
lamend = 2 * np.pi * C_LIGHT / start
lamstart = 2 * np.pi * C_LIGHT/stop
lam = np.arange(lamstart, lamend + resolution, resolution)
omega = 2 * np.pi * C_LIGHT / lam
relom = omega - center
_i = np.exp(-(relom) ** 2 / window)
i = 2 * _i + 2 * np.cos(relom * GD + (omega * delay)) * np.sqrt(_i * _i)
return omega, i
if __name__ == '__main__':
# Generate data
x, y = generate_data(1, 3, 2, 800, GD=0)
# Linearly interpolate to be evenly spaced
xs = np.linspace(x[0], x[-1], len(x))
intp = interp1d(x, y, kind='linear')
ys = intp(xs)
x, y = xs, ys
plt.plot(x, y, label='original')
# IFFT
xt = fftfreq(len(x), d=(x[0]-x[1])/(2*np.pi))
yt = ifft(y)
# plt.plot(xt, np.abs(yt))
# FFT back
xf = fftshift(fftfreq(len(xt), d=(xt[0]-xt[1])/(2*np.pi)))
yf = fft(yt)
plt.plot(xf, np.abs(yf), label='after transforms')
plt.legend()
plt.grid()
plt.show()
最佳答案
我认为fftfreq
不会做你认为它会做的事情。 xf
为 fft(ifft(y)
与 x
相同,您不应该尝试重新计算它。转到另一个域然后再返回时,x 轴不会改变。
另外,请注意 fftfreq
返回给定长度和给定样本间距的信号的离散傅立叶变换的频域坐标。它不会做相反的事情,在应用逆离散傅立叶变换后,您不能使用它来确定空间域中的坐标。 (它返回的间距有效,但坐标集无效。)
plt.plot(x, y, label='original')
# IFFT
yt = ifft(y)
# plt.plot(np.abs(yt))
# FFT back
yf = fft(yt)
plt.plot(x, np.real(yf), label='after transforms')
plt.legend()
plt.grid()
plt.show()
ifft(y)
假定沿 x 轴的一组固定值。您的
x
不符合这个。因此,您获得的空间域信号没有意义。
x
从 3.0 到 1.0,步长为 0.0004777。您必须增加数据,使值从 0.0 运行到 6.0,区域 (3.0, 6.0) 是区域 (0.0, 3.0) 的共轭对称副本。该区域对应于负频率,根据频域的周期性(F[n]==F[n+N],其中 N 为样本数)。用零填充区域 (0.0, 1.0)。
xf = fftfreq(len(xt), d=(xt[1]-xt[0]))
应该重建x轴。但是你需要计算
xt
适本地:
xt = np.linspace(0, 1/(x[1]-x[0]), len(x), endpoint=False)
(使用
x
标准化 DFT 频率轴)。
关于python - 在python中用FFT重建原始信号,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60213893/
有人可以向我解释该声明在C++中的含义吗?我从未见过这样的声明,只是对它的含义和作用感到困惑: int ix((dx > 0) - (dx > 1)); 最佳答案 您可以在括号或花括号中使用初始化程序
我有一个带有单词的mysql数据库。我用 while 语句打印所有单词。所以我觉得: 马铃薯番茄生菜 一切正常,但我想按长度对单词进行排序。我试过: if(strlen($go['words']) =
我忠实的路径遍历方法不再有效——它将空格视为分隔符。好久没做批处理编程了。 使用 FOR 循环时,唯一允许使用分隔符的是 FOR/F 选项。 我不想创建一个包含路径的临时文件,希望做如下的事情: C:
新建一个表: ? 1
我有一些带有多行块的文本文件,例如 2011/01/01 13:13:13,, Some Certain Text,=, [ certain text [
我想在 Vim 中文件的不同部分之间进行一些很好的分离: 我想用#'s 填充一行,然后在中间写上我的标题: ############################# 居中标题############
我该如何逃生 "*"至 "\*"在clojure?似乎无法让它工作: (s/replace "A*B" #"*" "*")生产 "A*B" (当然) (s/replace "A*B" #"*" "\*
这周我一直在努力更熟悉 C。我一直在阅读C Primer Plus (5th Edition) 但是我仍然在使用变量和指针时遇到了一些麻烦。 这是我用来测试的脚本: int main (int arg
在 Dart 中,初始化 List 有什么区别?使用 new 运算符并使用文字对其进行初始化? 情况1: List args = new List(2); args[0] = 1; args[1] =
我有一个字符向量,如下所示: "Internet" "Internet" "-1" "-5" "Internet" "Internet" 我想替换所有负数值的值(-1、-5 等
我有一个名为 gen 的数据框,如下所示 A B C D E 1 NA 4.35 35.3 3.36 4.8
我有一个字符向量,如下所示: "Internet" "Internet" "-1" "-5" "Internet" "Internet" 我想替换所有负数值的值(-1、-5 等
我想知道为什么 CMake 中的变量经常用美元符号和大括号括起来。例如,我看到这个电话in a CMake tutorial . include_directories(${PROJECT_BINAR
我正在尝试做这样的事情 $this->db->count_all("grant_money")->where('id',5); 这可能吗? 如果有任何其他方法可以做到这一点,请告诉我。谢谢 我想像上面
为什么这是有效的: int a = 5; int *aPtr = &a; printf("%i", *aPtr); 但这不是: int a = 5; int aPtr = &a; printf("%i
假设我有一个格式为“11.23.13”的日期字符串,我想用“/”替换每个点,使其看起来像“11/23/13”。 这是我的代码,但它无法正常工作,因为正则表达式看到“.”并将其解释为匹配每个字符而不是新
如何在键盘输入的字符处打印*? 例子: 如果我在控制台中输入:mouli,那么它应该将 m 替换为 *,然后是 o用 * 等等。 最佳答案 使用标准 API 无法解决此问题。如果这确实是一个明确的要求
我最近开始学习 Javascript,同时对卡在这段代码中的代码进行了一些实验: var k = { ab: "hi", func: function() { cons
我需要用“.”替换第一列中的重复项 例如: name1 name1 name1 name2 name2 name3 name3 我需要输出: name1 . . name2 . name3 . 我有这
我有以下两个表 education 和 jobs,每个表都有时间戳字段。在续集语句中,我想选择并确定两个表中保存的两个时间戳中哪个是最新的。 我已经尝试了以下但并不愉快; SELECT e.Sta
我是一名优秀的程序员,十分优秀!