Opencv实现傅里叶变换
作者:Ethan_Lei_Pro 发布时间:2023-08-24 18:53:29
傅里叶变换将图像分解成其正弦和余弦分量,它将图像由空域转换为时域。任何函数都可以近似的表示为无数正弦和余弦函数的和,傅里叶变换就是实现这一步的,数学上一个二维图像的傅里叶变换为:
公式中,f是图像在空域的值,F是频域的值。转换的结果是复数,但是不可能通过一个真实图像和一个复杂的图像或通过大小和相位图像去显示这样的一个图像。然而,在整个图像处理算法只对大小图像是感兴趣的,因为这包含了所有我们需要的图像几何结构的信息。
可通过以下几步显示一副傅里叶变换后的图像
1、将图像扩展到它的最佳尺寸,DFT(直接傅里叶变换)的性能依赖于图片的尺寸,当图像是2,3,5的倍数时往往是最快的。因此,为了达到最优性能通常采用垫边界值的方法,得到一个最佳的尺寸。
2、为傅立叶变换结果的实部和虚部分配存储空间。傅里叶变换的结果是一个复数,这意味着每幅图的结果都有一个实部和虚部,此外,频域范围远远大于它对应的空间范围。因此,我们这些通常至少以一个浮点数格式存储这些数值。因此,我们会将我们的输入图像转换为这种类型并且扩展它与另一通道存放复数值
3、进行傅里叶变换。
4、将复数转换为幅值,DFT的幅值由以下公式得出:
5、切换到对数刻度。对图像进行对数尺度的缩放,结果证明,傅立叶系数矩阵的动态范围太大,无法显示在屏幕上,我们无法通过这样去观察一些小的和高的变化值。因此那些高的数值将转化成白点而小的数值会变成黑点,使用灰度值进行可视化,我们可以将线性刻度转换为对数刻度,以便于观察。
6、剪切和重分布幅度图象,第一步我们扩展了图像,这里我们去掉扩展的那部分值,基于可视化的目的,我们还可以重新排列结果的象限,使原点(0,0)对应于与图像中心
7、归一化。目前得到的幅值图像仍然太大,超出了显示的范围,归一化这范围内的值,可以进一步达到可视化的目的
实现程序
void _DFT(){
//1以灰度模式读取原图像并显示
Mat srcImage = imread("miFan.jpg",0);
if (!srcImage.data){ cout << "Error\n"; }
imshow("原图像", srcImage);
//2将输入图像扩展到最佳尺寸,边界用0补充
int m = getOptimalDFTSize(srcImage.rows);
int n = getOptimalDFTSize(srcImage.cols);
//将添加的像素初始化为0
Mat padded;
copyMakeBorder(srcImage, padded, 0, m - srcImage.rows,
0, n - srcImage.cols, BORDER_CONSTANT, Scalar::all(0));
//3为傅里叶变换的结果(实部和虚部)分配存储空间
//将数组组合合并为一个多通道数组
Mat planes[] = { Mat_<float>(padded), Mat::zeros(padded.size(), CV_32F) };
Mat complexI;
merge(planes, 2, complexI);
//4进行傅里叶变换
dft(complexI, complexI);
//5将复数转换为幅值,即=> log(1 + sqrt(Re(DFT(I))^2 + Im(DFT(I))^2))
//将多通道数组分离为几个单通道数组
split(complexI, planes);//planes[0] = Re(DFT(I), planes[1] = Im(DFT(I))
magnitude(planes[0], planes[1], planes[0]);
Mat magImage = planes[0];
//6进行对数尺度缩放
magImage += Scalar::all(1);
log(magImage, magImage);//求自然对数
//7剪切和重分布幅度图象限
//若有奇数行或奇数列,进行频谱剪裁
magImage = magImage(Rect(0, 0, magImage.cols&-2, magImage.rows&-2));
//重新排列傅立叶图像中的象限,使得原点位于图像中心
int cx = magImage.cols / 2;
int cy = magImage.rows / 2;
Mat q0(magImage, Rect(0, 0, cx, cy));
Mat q1(magImage, Rect(cx, 0, cx, cy));
Mat q2(magImage, Rect(0,cy,cx,cy));
Mat q3(magImage, Rect(cx,cy,cx,cy));
//交换象限(左上与右下进行交换)
Mat tmp;
q0.copyTo(tmp);
q3.copyTo(q0);
tmp.copyTo(q3);
//交换象限(右上与左下进行交换)
q1.copyTo(tmp);
q2.copyTo(q1);
tmp.copyTo(q2);
//8归一化,用0到1的浮点值将矩阵变换为可视的图像格式
normalize(magImage, magImage, 0, 1, CV_MINMAX);
//9显示
imshow("频谱增幅", magImage);
waitKey();
}
傅里叶变换后的图片
来源:https://blog.csdn.net/baidu_17313961/article/details/49834981


猜你喜欢
- Java 匿名内部类详解匿名内部类也就是没有名字的内部类正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写但使用
- mybatis一直加载xml,找到错误我们在写springmvc+mybatis项目,启动项目的时候,mapper配置文件一直刷,一直加载。
- 背景之前和同事讨论一个问题,他们公司调研中发现forEach的速度比for的速度慢,当刚听到这个结论的时候有点诧异。因为之前看过国外的文章和
- Android RadioButton 图片位置与大小Java:rgGroup = (RadioGroup) findViewById(R.
- 做Android开发两年的时间,技术稍稍有一些提升,刚好把自己实现的功能写出来,记录一下,如果能帮助到同行的其他人,我也算是做了件好事,哈哈
- Map接口Map提供了一种映射关系,其中的元素是以键值对(key-value)的形式存储的,能够实现根据key快速查找value;Map中的
- 本文实例为大家分享了Android简单实现文件下载的具体代码,供大家参考,具体内容如下权限<!-- 文件读写权限 &nbs
- 本文实例为大家分享了C语言实现超市计价收款系统的具体代码,供大家参考,具体内容如下学习一个月的C语言,写了一个简易的超市计价收款系统源码如下
- 在某种场景下,可能我们需要获取app的图标名称和启动图片的名称。比如说app在前台时,收到了远程通知但是通知栏是不会有通知提醒的,这时我想做
- 本文实例为大家分享了Android实现ListView下拉刷新上拉加载更多的具体代码,供大家参考,具体内容如下其实谷歌官方目前已经推出Lis
- 前言在我们平时使用图形化界面的时候,会发现来建立一个文件夹或者一个文档的时候很简单,只需要在桌面单击鼠标右键就可以了。但是,在我们写项目的时
- 甲:听说最近java跌落神坛,python称霸武林了,你知道吗?乙:不是吧,我前几天看python怎么还是第三?丙:你们都在扯蛋,pytho
- 本文实例讲述了java获取百度网盘真实下载链接的方法。分享给大家供大家参考。具体如下:目前还存在一个问题,同一ip在获取3次以后会出现验证码
- 本文实例讲述了Android编程将Activity背景设置为墙纸的简单实现方法。分享给大家供大家参考,具体如下:1)代码方式Drawable
- 鼠标事件监听机制的三个方面:1.事件源对象:事件源对象就是能够产生动作的对象。在Java语言中所有的容器组件和元素组件都是事件监听中的事件源
- 本想把练习题做了的结果放上来,不过发现附录是有答案的,就算了吧,自己做了没问题就行了哈。之前提到过,要是有朋友有想法,需要做小工具我可以帮忙
- 补充知识:正定矩阵奇异矩阵严格对角占优要理解Gauss消去法,首先来看一个例子:从上例子可以看出,高斯消去法实际上就是我们初中学的阶二元一次
- 本文实例介绍的是Android的Tab控件,Tab控件可以达到分页的效果,让一个屏幕的内容尽量丰富,当然也会增加开发的复杂程度,在有必要的时
- 目录:1.list中添加,获取,删除元素;2.list中是否包含某个元素;3.list中根据索引将元素数值改变(替换);4.list中查看(
- 其实SynchronousQueue 是一个特别有意思的阻塞队列,就我个人理解来说,它很重要的特点就是没有容量。直接看一个例子:packag