基于MFC和opencv的FFT
在网上折腾了一阵子,终于把这个程序写好了,程序是基于MFC的,图像显示的部分和获取图像的像素点是用到了opencv的一些函数,不过FFT算法没有用opencv的(呵呵,老师不让),网上的二维的FFT程序一般都是把图像分别进行行变换后进行列变换的,在编程过程中遇到了一些问题,是这样的,FFT算法算完后得到的复数矩阵怎么imshow?问题就出现在这,我原来的程序因为归一化到0-255时,程序运行特别慢(用了个CArray,找出array里的最大值和最小值,然后(每一个复数矩阵求模后-最小值)/(最大值-最小值),不满才怪呵呵,得出FFT结果是全黑的)。参考了别人的归一化
/*-----------------------------------------------------------------------------
* 计算功率谱
* 和归一化
*
*-----------------------------------------------------------------------------*/
// 行
for(i = ; i < h; i++)
{
// 列
for(j = ; j < w; j++)
{
// 计算频谱
dTemp = sqrt(FD[j * h + i].real() * FD[j * h + i].real() +
FD[j * h + i].imag() * FD[j * h + i].imag()) / ; // 判断是否超过255
if (dTemp > )
{
// 对于超过的,直接设置为255
dTemp = ;
} image.at<uchar>(i,j)=(uchar)dTemp;
// 更新源图像 }
}
主要代码:
/*
* =====================================================================================
*
* Filename: fft_dlgDlg.cpp
* Environment:opencv2.4.4 vs2010
*
* Description: 基于MFC的FFT程序
*
*
*
* Version: 1.0
* Created: 2013/10/19 19:24:06
* Author: yuliyang
I*
* Mail: wzyuliyang911@gmail.com
* Blog: http://www.cnblogs.com/yuliyang
*
* =====================================================================================
*/ // fft_dlgDlg.cpp : 实现文件
// #include "stdafx.h"
#include "fft_dlg.h"
#include "fft_dlgDlg.h"
#include "afxdialogex.h" #include <complex>
#include <math.h>
#include <stdio.h>
#include <Windows.h>
#include "opencv\highgui.h"
#include "opencv2/core/core.hpp"
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
using namespace cv;
using namespace std;
#define PI 3.1415936; #ifdef _DEBUG
#define new DEBUG_NEW
#endif // 用于应用程序“关于”菜单项的 CAboutDlg 对话框 class CAboutDlg : public CDialogEx
{
public:
CAboutDlg(); // 对话框数据
enum { IDD = IDD_ABOUTBOX }; protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV 支持 // 实现
protected:
DECLARE_MESSAGE_MAP()
}; CAboutDlg::CAboutDlg() : CDialogEx(CAboutDlg::IDD)
{
} void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
} BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP() // Cfft_dlgDlg 对话框 Cfft_dlgDlg::Cfft_dlgDlg(CWnd* pParent /*=NULL*/)
: CDialogEx(Cfft_dlgDlg::IDD, pParent)
{
m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
} void Cfft_dlgDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
} BEGIN_MESSAGE_MAP(Cfft_dlgDlg, CDialogEx)
ON_WM_SYSCOMMAND()
ON_WM_PAINT()
ON_WM_QUERYDRAGICON()
ON_BN_CLICKED(IDC_OPEN_FILE, &Cfft_dlgDlg::OnBnClickedOpenFile) END_MESSAGE_MAP() // Cfft_dlgDlg 消息处理程序 BOOL Cfft_dlgDlg::OnInitDialog()
{
CDialogEx::OnInitDialog(); // 将“关于...”菜单项添加到系统菜单中。 // IDM_ABOUTBOX 必须在系统命令范围内。
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
ASSERT(IDM_ABOUTBOX < 0xF000); CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != NULL)
{
BOOL bNameValid;
CString strAboutMenu;
bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
ASSERT(bNameValid);
if (!strAboutMenu.IsEmpty())
{
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
}
} // 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
// 执行此操作
SetIcon(m_hIcon, TRUE); // 设置大图标
SetIcon(m_hIcon, FALSE); // 设置小图标 // TODO: 在此添加额外的初始化代码 return TRUE; // 除非将焦点设置到控件,否则返回 TRUE
} void Cfft_dlgDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
}
else
{
CDialogEx::OnSysCommand(nID, lParam);
}
} // 如果向对话框添加最小化按钮,则需要下面的代码
// 来绘制该图标。对于使用文档/视图模型的 MFC 应用程序,
// 这将由框架自动完成。 void Cfft_dlgDlg::OnPaint()
{
if (IsIconic())
{
CPaintDC dc(this); // 用于绘制的设备上下文 SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), ); // 使图标在工作区矩形中居中
int cxIcon = GetSystemMetrics(SM_CXICON);
int cyIcon = GetSystemMetrics(SM_CYICON);
CRect rect;
GetClientRect(&rect);
int x = (rect.Width() - cxIcon + ) / ;
int y = (rect.Height() - cyIcon + ) / ; // 绘制图标
dc.DrawIcon(x, y, m_hIcon);
}
else
{
CDialogEx::OnPaint();
}
} //当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR Cfft_dlgDlg::OnQueryDragIcon()
{
return static_cast<HCURSOR>(m_hIcon);
} void Cfft_dlgDlg::OnBnClickedOpenFile()
{
// TODO: 在此添加控件通知处理程序代码 /*-----------------------------------------------------------------------------
* 选择文件名
*-----------------------------------------------------------------------------*/ CString strFileName,strszFilter,strtitle,strext;
strszFilter="位图文件(*.bmp)|*.bmp|位图文件(*.jpg)|*.jpg|全部文件(*.*)|*.*||";
CFileDialog bmpdlg(TRUE,NULL,NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,strszFilter,NULL);
if(IDOK == bmpdlg.DoModal())
{
strFileName = bmpdlg.GetPathName();
strtitle=bmpdlg.GetFileTitle();
strext=bmpdlg.GetFileExt(); }
if (strFileName.IsEmpty())
{
//MessageBox((LPCTSTR)strFileName);
MessageBox("请选择一副图像"); return ;
} /*-----------------------------------------------------------------------------
* opencv读入图像
*
*
*-----------------------------------------------------------------------------*/
Mat image=imread(strFileName.GetBuffer(),); //unsigned char* lpSrc;
// 中间变量
double dTemp;
// 循环变量
LONG i;
LONG j;
// 进行付立叶变换的宽度和高度(2的整数次方)
LONG w;
LONG h;
int wp;
int hp;
// 赋初值
w = ;
h = ;
wp = ;
hp = ; /*-----------------------------------------------------------------------------
* 填充到2的幂次方
*
*
*-----------------------------------------------------------------------------*/
// 计算进行付立叶变换的宽度和高度(2的整数次方)
while(w * <= image.cols)
{
w *= ;
wp++;
} while(h * <= image.rows)
{
h *= ;
hp++;
}
// 分配内存
complex<double> *TD = new complex<double>[w * h];
complex<double> *FD = new complex<double>[w * h];
// 行
for(i = ; i < h; i++)
{
// 列
for(j = ; j < w; j++)
{
// 给时域赋值
TD[j + w * i] = complex<double>(image.at<uchar>(i,j), ); /* opencv函数读取图像像素到复数矩阵里 */
}
}
/*-----------------------------------------------------------------------------
* 把二维的FFT换成分别对行方向和列方向进行一维的FFT
*
*
*-----------------------------------------------------------------------------*/
for(i = ; i < h; i++)
{
// 对y方向进行快速付立叶变换
FFT(&TD[w * i], &FD[w * i], wp);
}
// 保存变换结果
for(i = ; i < h; i++)
{
for(j = ; j < w; j++)
{
TD[i + h * j] = FD[j + w * i];
}
} for(i = ; i < w; i++)
{
// 对x方向进行快速付立叶变换
FFT(&TD[i * h], &FD[i * h], hp);
} /*-----------------------------------------------------------------------------
* 计算功率谱
* 和归一化
*
*-----------------------------------------------------------------------------*/
// 行
for(i = ; i < h; i++)
{
// 列
for(j = ; j < w; j++)
{
// 计算频谱
dTemp = sqrt(FD[j * h + i].real() * FD[j * h + i].real() +
FD[j * h + i].imag() * FD[j * h + i].imag()) / ; // 判断是否超过255
if (dTemp > )
{
// 对于超过的,直接设置为255
dTemp = ;
} image.at<uchar>(i,j)=(uchar)dTemp;
// 更新源图像 }
} /*-----------------------------------------------------------------------------
* 释放内存
*
*
*-----------------------------------------------------------------------------*/
// 删除临时变量
delete TD;
delete FD; /*-----------------------------------------------------------------------------
* 进行图像的中心化
*
*
*-----------------------------------------------------------------------------*/
int cy = image.rows/; /* 中心点位置 cx ,cy */
int cx = image.cols/;
uchar tmp13,tmp24; //imshow("未中心化的功率谱",image); IplImage center_image=IplImage(image);
for( j = ; j < cy; j++ ){
for( i = ; i < cx; i++ ){
//中心化,将整体份成四块进行对角交换
tmp13 = CV_IMAGE_ELEM( ¢er_image, uchar, j, i);
CV_IMAGE_ELEM( ¢er_image, uchar, j, i) = CV_IMAGE_ELEM(
¢er_image, uchar, j+cy, i+cx);
CV_IMAGE_ELEM( ¢er_image, uchar, j+cy, i+cx) = tmp13; tmp24 = CV_IMAGE_ELEM( ¢er_image, uchar, j, i+cx);
CV_IMAGE_ELEM( ¢er_image, uchar, j, i+cx) =
CV_IMAGE_ELEM( ¢er_image, uchar, j+cy, i);
CV_IMAGE_ELEM( ¢er_image, uchar, j+cy, i) = tmp24;
}
} /*-----------------------------------------------------------------------------
* 用于保存FFT图像
*
*
*-----------------------------------------------------------------------------*/
Mat img(¢er_image,);
if (BST_CHECKED == ::IsDlgButtonChecked(m_hWnd,IDC_CHECK_SAVE))
{ strtitle="FFT_"+strtitle+"."+strext; //MessageBox(strtitle.GetBuffer(0));
imwrite(strtitle.GetBuffer(),img); }
imshow("中心化后的功率谱",img); waitKey();
// 返回
//return TRUE; } /*
* === FUNCTION ======================================================================
* Name: FFT
* Description: 计算一维FFT
* =====================================================================================
*/
void Cfft_dlgDlg::FFT(complex<double> * TD, complex<double> * FD, int r)
{ LONG count;
// 循环变量
int i,j,k;
// 中间变量
int bfsize,p;
// 角度
double angle;
complex<double> *W,*X1,*X2,*X;
// 计算付立叶变换点数
count = << r;
// 分配运算所需存储器
W = new complex<double>[count / ];
X1 = new complex<double>[count];
X2 = new complex<double>[count];
// 计算加权系数
for(i = ; i < count / ; i++)
{
angle =-*i*PI;
angle =angle/(double)count;
W[i] = complex<double> (cos(angle), sin(angle));
}
// 将时域点写入X1
memcpy(X1, TD, sizeof(complex<double>) * count); /*-----------------------------------------------------------------------------
*
* 采用蝶形算法进行快速付立叶变换
*
*-----------------------------------------------------------------------------*/ for(k = ; k < r; k++)
{
for(j = ; j < << k; j++)
{
bfsize = << (r-k);
for(i = ; i < bfsize / ; i++)
{
p = j * bfsize;
X2[i + p] = X1[i + p] + X1[i + p + bfsize / ];
X2[i + p + bfsize / ] = (X1[i + p] - X1[i + p + bfsize / ]) * W[i * (<<k)];
}
}
X = X1;
X1 = X2;
X2 = X;
}
// 重新排序
for(j = ; j < count; j++)
{
p = ;
for(i = ; i < r; i++)
{
if (j&(<<i))
{
p+=<<(r-i-);
}
}
FD[j]=X1[p];
}
// 释放内存
delete W;
delete X1;
delete X2;
}
运行效果如下:
提供程序一份:
http://pan.baidu.com/s/1ehvwy
基于MFC和opencv的FFT的更多相关文章
- 转:基于开源项目OpenCV的人脸识别Demo版整理(不仅可以识别人脸,还可以识别眼睛鼻子嘴等)【模式识别中的翘楚】
文章来自于:http://blog.renren.com/share/246648717/8171467499 基于开源项目OpenCV的人脸识别Demo版整理(不仅可以识别人脸,还可以识别眼睛鼻子嘴 ...
- 基于MFC的socket编程(异步非阻塞通信)
对于许多初学者来说,网络通信程序的开发,普遍的一个现象就是觉得难以入手.许多概念,诸如:同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)等,初学者往往迷惑不清, ...
- 基于MFC简单图片裁剪工具
话说这几天又没干啥正事,看书没效率,游戏也没怎么玩,尼玛时间都去哪儿了! --------------------------------------------------------------- ...
- windows平台下基于QT和OpenCV搭建图像处理平台
在之前的博客中,已经分别比较详细地阐述了"windows平台下基于VS和OpenCV"以及"Linux平台下基于QT和OpenCV"搭建图像处理框架,并 ...
- 基于MFC开发的指纹识别系统.
MFC-FingerPrint 基于MFC开发的指纹识别系统. 效果图如下: 在第12步特征入库中,会对当前指纹的mdl数据与databases中所有的mdl进行对比,然后返回识别结果. 一.载入图像 ...
- 基于python的快速傅里叶变换FFT(二)
基于python的快速傅里叶变换FFT(二)本文在上一篇博客的基础上进一步探究正弦函数及其FFT变换. 知识点 FFT变换,其实就是快速离散傅里叶变换,傅立叶变换是数字信号处理领域一种很重要的算法. ...
- 最全的基于MFC的ActiveX控件开发教程
浏览器插件之ActiveX开发(一) 一般的Web应用对于浏览器插件能不使用的建议尽量不使用,因为其涉及到安全问题以及影响用户安装(或自动下载注册安装)体验问题.在有特殊需求(如涉及数据安全的金融业务 ...
- 基于MFC的ActiveX控件开发教程------------浏览器插件之ActiveX开发
浏览器插件之ActiveX开发(一) 一般的Web应用对于浏览器插件能不使用的建议尽量不使用,因为其涉及到安全问题以及影响用户安装(或自动下载注册安装)体验问题.在有特殊需求(如涉及数据安全的金融业务 ...
- Beginning SDL 2.0(5) 基于MFC和SDL的YuvPlayer
本文是在“Beginning SDL 2.0(4) YUV加载及渲染”(以下简称BS4)基础上做的功能完善,如果你对之间介绍的内容了解不多,麻烦先阅读之前的内容. 本文主要介绍如何完成一个基于MFC和 ...
随机推荐
- Sina App Engine(SAE)入门教程(6)- memcache使用
Memcache是一个高性能的分布式的内存对象缓存系统,通过在内存里维护一个统一的巨大的hash表,它能够用来存储各种格式的数据,包括图像.视频.文件以及数据库检索的结果等.简单的说就是将数据调用到内 ...
- 基于Struts2框架实现登录案例 之 程序国际化
国际化牵涉的知识非常多,这里只能简单的介绍,程序国际化的一般做法是:在jsp页面时, 不是直接输出信息,而是输出一个key值,该key值在不同语言环境下找到对应资源文件下的 对应信息,因此首先要创建满 ...
- C++类型转化
dynamic_cast在多继承中可以判断无关类之间是否可以转换,例如基类B1和基类B2之间,static_cast可以在隐式转化及其逆转换中进行,但不允许无关类型之间的转换.,dynamic_cas ...
- 在BSP的.bat文件下設置全局變量方法
用于多個產品共用一個BSP的時候,在BSP的.bat文件中設置全局變量,去掉不需要加載的驅動和不同點是很好的方法. 一,舉例:BSP中.bat的一段code: set BSP_SMDK2443=1 s ...
- Java面向对象详解
Java面向对象详解 前言:接触项目开发也有很长一段时间了,最近开始萌发出想回过头来写写以前学 过的基础知识的想法.一是原来刚开始学习接触编程,一个人跌跌撞撞摸索着往前走,初学的时候很多东西理解的也懵 ...
- TeeChart的X轴为时间,多个Y轴的显示
最后上代码 public partial class Test : Form { private TChart tChart = new TChart(); ; public Test() { Ini ...
- ER模型到关系模型的转换规则
E-R模型向关系模型的转换规则: 一.两元联系的转换规则 (1)实体类型的转换 将每个实体类型转换成一个关系模式,实体的属性即为关系的属性,实体标识符即为关系的键. (2)联系类型的转换 a实体间的联 ...
- C++ 中字符串标准输入的学习及实验
声明:下面实验中[]里面表示要输入里面的符号,[]符号本身并未输入 1.cin>> cin使用空白(空格.制表符.回车)来确定字符串的结束位置. cin会将换行符留在输入输出队列中. #i ...
- IntelliJ IDEA For Mac 快捷键——常用版
一.搜索 搜索文件 command+shift+n 打开方法实现类 command+option+b 全文搜索 ctrl+shift+f (1)类和方法 查看类的继承结构 ctrl+h 查看方法的 ...
- How can I add a new user as sudoer using the command line?
Two ways to use sudo command for a standard user account: First, If you want to use sudo command for ...