一、

1) 将宽为2n的正方形图像,用FFT算法从空域变换到频域,并用频域图像的模来进行显示。

2) 使图像能量中心,对应到几何中心,并用频域图像的模来进行显示。

3)将频域图象,通过FFT逆变换到空域,并显示。

#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
#include <iostream>
using namespace cv;
using namespace std; int main()
{ //以灰度模式读取原始图像并显示
Mat srcImage = imread("lena.png", 0);
if (srcImage.empty())
{
cout << "打开图像失败!" << endl;
return -1;
}
imshow("原始图像", srcImage); //将输入图像延扩到最佳的尺寸,边界用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)); //为傅立叶变换的结果(实部和虚部)分配存储空间。
//将planes数组组合合并成一个多通道的数组complexI
Mat planes[] = { Mat_<float>(padded), Mat::zeros(padded.size(), CV_32F) };
Mat complexI;
merge(planes, 2, complexI); //进行就地离散傅里叶变换
dft(complexI, complexI); //将复数转换为幅值,即=> log(1 + sqrt(Re(DFT(I))^2 + Im(DFT(I))^2))
split(complexI, planes); // 将多通道数组complexI分离成几个单通道数组,planes[0] = Re(DFT(I), planes[1] = Im(DFT(I))
magnitude(planes[0], planes[1], planes[0]);// planes[0] = magnitude
Mat magnitudeImage = planes[0]; //进行对数尺度(logarithmic scale)缩放
magnitudeImage += Scalar::all(1);
log(magnitudeImage, magnitudeImage);//求自然对数 //剪切和重分布幅度图象限
//若有奇数行或奇数列,进行频谱裁剪
magnitudeImage = magnitudeImage(Rect(0, 0, magnitudeImage.cols & -2, magnitudeImage.rows & -2));
//重新排列傅立叶图像中的象限,使得原点位于图像中心
int cx = magnitudeImage.cols / 2;
int cy = magnitudeImage.rows / 2;
Mat q0(magnitudeImage, Rect(0, 0, cx, cy)); // ROI区域的左上
Mat q1(magnitudeImage, Rect(cx, 0, cx, cy)); // ROI区域的右上
Mat q2(magnitudeImage, Rect(0, cy, cx, cy)); // ROI区域的左下
Mat q3(magnitudeImage, Rect(cx, cy, cx, cy)); // ROI区域的右下
//交换象限(左上与右下进行交换)
Mat tmp;
q0.copyTo(tmp);
q3.copyTo(q0);
tmp.copyTo(q3);
//交换象限(右上与左下进行交换)
q1.copyTo(tmp);
q2.copyTo(q1);
tmp.copyTo(q2); //归一化,用0到1之间的浮点值将矩阵变换为可视的图像格式
normalize(magnitudeImage, magnitudeImage, 0, 1, CV_MINMAX); //显示效果图
imshow("频域", magnitudeImage); //频域-->空域
Mat inversed;
dft(complexI, inversed, DFT_INVERSE | DFT_REAL_OUTPUT);
normalize(inversed, inversed, 0, 1, CV_MINMAX);
imshow("空域", inversed); waitKey(); return 0;
}

二、

对于下面这幅图像,请问可以通过那些图像增强的手段,达到改善视觉效果的目的?请显示处理结果,并附简要处理流程说明。

#include <opencv2/opencv.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
#include <iostream>
using namespace std;
using namespace cv;
int ContrastValue; //对比度值
int BrightValue; //亮度值
Mat src, dst;
//改变图像对比度和亮度值的回调函数
static void ContrastAndBright(int, void *)
{
//创建窗口
namedWindow("【原始图窗口】", WINDOW_AUTOSIZE);
for (int y = 0; y < src.rows; y++)
{
for (int x = 0; x < src.cols; x++)
{
for (int c = 0; c < 3; c++)
{
dst.at<Vec3b>(y, x)[c] = saturate_cast<uchar>((ContrastValue * 0.01)*(src.at<Vec3b>(y, x)[c]) + BrightValue);
}
}
}
//显示图像
imshow("【原始图窗口】", src);
imshow("【效果图窗口】", dst);
} int main(int argc, char *argv[])
{
//打开图像
src = imread("two.png");
if (src.empty())
{
cout << "打开图像失败!" << endl;
return -1;
}
//中值滤波去噪
medianBlur(src, src, 3);
dst = Mat::zeros(src.size(), src.type());
//设定对比度和亮度的初值
ContrastValue = 80;
BrightValue = 80;
//创建窗口
namedWindow("【效果图窗口】", WINDOW_AUTOSIZE);
//创建轨迹条
createTrackbar("对比度:", "【效果图窗口】", &ContrastValue, 300, ContrastAndBright);
createTrackbar("亮 度:", "【效果图窗口】", &BrightValue, 200, ContrastAndBright);
//调用回调函数
ContrastAndBright(ContrastValue, 0);
ContrastAndBright(BrightValue, 0);
//等待用户按键,起到暂停的作用
waitKey();
return 0;
}

三、

对于下面这幅图像,编程实现染色体计数,并附简要处理流程说明。

#include <opencv2/opencv.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
#include <iostream>
#include <vector>
using namespace std;
using namespace cv;
int main(int argc, char **argv)
{
Mat gray, src, dst;
//打开图像
src = imread("image.png");
if (src.empty())
{
cout << "打开图像失败!" << endl;
return -1;
}
cout << "rows = " << src.rows << endl;
cout << "cols = " << src.cols << endl;
//转换为灰度图
cvtColor(src, gray, CV_BGR2GRAY);
//中值滤波
medianBlur(gray, gray, 7);
//图像二值化
threshold(gray, dst, 170, 255, THRESH_BINARY);
//腐蚀,默认内核3*3
erode(dst, dst, Mat());
//erode(dst, dst, Mat()); Mat canny_output;
vector<vector<Point> > contours;
vector<Vec4i> hierarchy;
//画轮廓线
Canny(dst, canny_output, 100, 100 * 2, 3);
imwrite("data.png", dst); //检测轮廓
findContours(dst, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0));
cout << "一共检测到染色体数目:" << contours.size() - 1 << endl;
/*
for (int i = 0; i < contours.size(); i++)
{
for (int j = 0; j < contours[i].size(); j++)
{
cout << contours[i][j] << " ";
}
cout << ";" << endl;
}
*/ //显示图片
imshow("src", src);
imshow("canny_output", canny_output); //将图片保存到文件
imwrite("dst.png", canny_output);
//等待用户输入
waitKey();
return 0;
} //高斯滤波
//GaussianBlur(gray, gray, Size(5, 5), 0, 0); //双边滤波
//bilateralFilter(gray, gray, 5, 10.0, 2.0); //中值滤波
//medianBlur(gray, gray, 3);

四、

对MNIST手写数字数据库(可在网上搜索下载),编程实现来提取其链码。
#include <opencv2/opencv.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>
#include <iostream>
#include <vector>
#include <string>
#include <fstream> using namespace std;
using namespace cv; //格式转换
int ReverseInt(int i)
{
unsigned char ch1, ch2, ch3, ch4;
ch1 = i & 255;
ch2 = (i >> 8) & 255;
ch3 = (i >> 16) & 255;
ch4 = (i >> 24) & 255;
return((int)ch1 << 24) + ((int)ch2 << 16) + ((int)ch3 << 8) + ch4;
} /**
* 将Mnist数据库读取到OpenCV::Mat格式中
* 格式:
* magic number
* number of images
* rows
* cols
* a very very long vector contains all digits
*/
void read_Mnist(string filename, vector<Mat> &vec)
{
ifstream file(filename, ios::binary);
if (file.is_open())
{
int magic_number = 0;
int number_of_images = 0;
int n_rows = 0;
int n_cols = 0;
file.read((char*)&magic_number, sizeof(magic_number));
magic_number = ReverseInt(magic_number); file.read((char*)&number_of_images, sizeof(number_of_images));
number_of_images = ReverseInt(number_of_images); file.read((char*)&n_rows, sizeof(n_rows));
n_rows = ReverseInt(n_rows); file.read((char*)&n_cols, sizeof(n_cols));
n_cols = ReverseInt(n_cols); for (int i = 0; i < number_of_images; ++i)
{
cv::Mat tp = Mat::zeros(n_rows, n_cols, CV_8UC1);
for (int r = 0; r < n_rows; ++r)
{
for (int c = 0; c < n_cols; ++c)
{
unsigned char temp = 0;
file.read((char*)&temp, sizeof(temp));
tp.at<uchar>(r, c) = (int)temp;
}
}
vec.push_back(tp);
}
}//if
} int main(int argc, char **argv)
{
int count = 1;
//存储Mnist字库
vector<Mat> vec;
//将Mnist字库读取到vector中
read_Mnist("t10k-images.idx3-ubyte", vec);
cout << "共含有:" << vec.size() << "幅图片" << endl; for (auto iter = vec.begin(); iter != vec.end(); iter++)
{
cout << "第" << count++ << "幅图片..." << endl;
//显示Mnist字库
imshow("Mnist", *iter);
vector<vector<Point> > contours; //读取轮廓
findContours(*iter, contours, CV_RETR_EXTERNAL, CV_CHAIN_CODE);
//输出链码
for (int i = 0; i < contours.size(); i++)
{
for (int j = 0; j < contours[i].size(); j++)
{
cout << contours[i][j];
}
cout << endl;
}
contours.clear();
waitKey(1000);
}
waitKey();
return 0;
}

详细说明:

BJUT数字图像处理作业的更多相关文章

  1. 数字图像处理作业使用OpenCV - 使用笔记

    数字图像处理作业的输入图像全部都是灰度图像,所以汇总一下自己遇到的问题答案. OCV的图像容器是Mat<typename>,可以用imread(filename)读取图像,filename ...

  2. 数字图像处理作业使用OpenCV - 块提取

    今天要记录的是树图第二次作业的第二题,Image Patch Extraction.这个概念真的不难懂,但是如果要我实际写的话,还真的不知道要怎么去遍历图像矩阵来提取块.在此要多谢邓大神的热心帮助,告 ...

  3. 数字图像处理作业使用OpenCV - 自定义直方图

    第二次作业需要打印出来灰度直方图,当然不能使用ocv的自带calcHist函数来得到Mat对象了……结果上网搜索怎么用自己的数据创建直方图,搜到的都是直接用函数的_(:з」∠)_ 结果这个地方拖了好久 ...

  4. 数字图像处理作业使用OpenCV - 配置

    使用环境:Windows7 旗舰版 + vs2008 + OpenCV2.0a 基本上配置都是通过网上一个教程,在此附上地址 Click ME. 为了避免因不同版本而出现的安装问题,我还是下载了2.0 ...

  5. 数字图像处理(一)之灰度转换和卷积python实现

    使用Python实现数字图像处理中如下功能: 彩色图像转成灰度图像 实现图像的相关&卷积操作 实现图像的高斯核卷积 使用的库和python版本如下: imageio:2.9.0 用于读取磁盘中 ...

  6. 基于小波变换的数字图像处理(MATLAB源代码)

    基于小波变换的数字图像处理(MATLAB源代码) clear all; close all; clc;M=256;%原图像长度N=64; %水印长度[filename1,pathname]=uiget ...

  7. python数字图像处理(17):边缘与轮廓

    在前面的python数字图像处理(10):图像简单滤波 中,我们已经讲解了很多算子用来检测边缘,其中用得最多的canny算子边缘检测. 本篇我们讲解一些其它方法来检测轮廓. 1.查找轮廓(find_c ...

  8. python数字图像处理(1):环境安装与配置

    一提到数字图像处理编程,可能大多数人就会想到matlab,但matlab也有自身的缺点: 1.不开源,价格贵 2.软件容量大.一般3G以上,高版本甚至达5G以上. 3.只能做研究,不易转化成软件. 因 ...

  9. 数字图像处理之sobel边缘检测

    在前两部文章介绍了几种边缘检测算法,和位图的内存结构.如果对前两篇文章已经理解透彻 了,那么本文将带你进入数字图像处理的世界. 本文通过C代码实现基本的sobel边缘检测,包括8个方向和垂直方向: 代 ...

随机推荐

  1. 使用EF操作Docker中的Mysql实例

    为啥我会选择mysql呢?因为我的服务器配置较低,而SqlServer在docker中的实例,服务器的运行内存应当保持在2G+,我没有这个条件,它会爆出这样的错误 sqlservr: This pro ...

  2. 大数据学习之路-Centos6安装python3.5

    Centos 6.8安装python3.5.2 因为学习所需,需要用到python3.x的环境,目前Linux系统默认的版本都是python2.x的,还有一些自带的工具需要用到python2.6版本, ...

  3. (二)Django模板的应用

    一.配置项目的基础模板,分三部分 base.html 基础部分,需要包括基本的HTML标签 header部分 body部分 html闭合标签 {% load staticfiles %} <!D ...

  4. 简单聊一聊JS中的循环引用及问题

    本文主要从 JS 中为什么会出现循环引用,垃圾回收策略中引用计数为什么有很大的问题,以及循环引用时的对象在使用 JSON.stringify 时为什么会报错,怎样解决这个问题简单谈谈自己的一些理解. ...

  5. springboot多环境(dev,test,prod)配置

    前情提要 在我们开发工作中,常常因为配置的问题,搞得头昏脑大.开发环境.测试环境.配置各不相同,数据库.redis.注册中心等等参数都不一致,如果放在同一个配置文件,就会发现诸多注释,发布不同的环境, ...

  6. js实现类选择器和name属性选择器

    jQuery的出现,大大的提升了我们操作dom的效率,使得我们的开发更上一层楼,如jQuery的选择器就是一个很强大的功能,它包含了类选择器.id选择器.属性选择器.元素选择器.层级选择器.内容筛选选 ...

  7. 越来越清晰的TFRecord处理图片的步骤

    # 首先是模块的导入 """ os模块是处理文件夹用的 PIL模块是用来处理图片的 """ import tensorflow as tf ...

  8. Java Collection集合概述及其常用方法

    Collection集合概述 Java数组的长度是固定的,为了使程序能够方便地存储和操作数目不固定的一组数据,JDK类库提供了Java集合 与数组不同的是,集合中不能存放基本类型数据,而只能存放对象的 ...

  9. 0x80070035找不到网络路径

    如果这个报错发生:自己的网络正常,其他人可以正常访问服务器,而自己无法访问服务器.原因就是TCP/IP NetBIOS Helper服务被停止. 打开services.msc,启动此服务即可. 该服务 ...

  10. pip 下载源更换

    Python博大精深之处在于丰富的库,而目前最方便的下载库的方法无疑是pip.但是由于国内的网络环境导致,好多库下载是在太慢了,原因大家都懂得.. 一.临时修改 使用pip的时候加入参数 -i pip ...