【图像处理】利用C++编写函数,绘制灰度图像直方图
1. 简介
利用OpenCV读取图像,转换为灰度图像,绘制该灰度图像直方图。点击直方图,控制台输出该灰度级像素个数。
2. 原理
(1) 实现原理较为简单,主要利用了OpenCV读取图像,并转换为灰度图像;
srcImg = imread(" ......"); // “....” 代表图像地址
if (srcImg.empty()) {
return -;
}
imshow(WINDOW_SRCIMG, srcImg);
Mat grayImg;
cvtColor(srcImg, grayImg, CV_BGR2GRAY);
imshow(WINDOW_GRAYIMG, grayImg);
(2) 利用Mat类新建一个固定分辨率的画布,统计处于每一灰度级像素个数,在该画布上绘制灰度直方图。
int nRows = ,nCols=;
Mat g_dstImg(nRows,nCols, CV_8UC1, Scalar::all()); // 新建画布
同时避免画布中该灰度级太高而超出画布范围,在本程序中采用了等比例缩小的方法。
int MaxCount = arrayMax(grayLevel,);//寻找在处于某一灰度级中个数最多的像素个数
yscaleRate = double(nRows)/MaxCount ;//y缩放比例
double xscaleRate = nCols / ;//x缩放比例
3. 实施细节
// 灰度直方图.cpp : 定义控制台应用程序的入口点。
// Topic: 绘制灰度图的直方图; 转载请注明出处:Chen_HW (https://www.cnblogs.com/chen-hw/p/11668119.html)
// Env: VS2015 + Debug x64 + OpenCV3.4.1
// Date:2018.11.22 by Chen_HW
#include "stdafx.h"
#include <opencv2/opencv.hpp>
#include <iostream>
#define WINDOW_SRCIMG "【源图】"
#define WINDOW_GRAYIMG "【灰度图】"
#define WINDOW_HIST "【直方图】"
using namespace std;
using namespace cv;
int arrayMax(int g_arr[], int num);
Mat drawHist(Mat &g_srcImg);
void MouseHandle(int event, int x, int y, int flags, void *param);
Point clickPoint,displayPoint;
bool downFlag = false;
Mat srcImg;
double yscaleRate = ; int main()
{
system("color 3f");
srcImg = imread(" ......"); // “....” 代表图像地址
if (srcImg.empty()) {
return -;
}
imshow(WINDOW_SRCIMG, srcImg);
Mat grayImg;
cvtColor(srcImg, grayImg, CV_BGR2GRAY);
imshow(WINDOW_GRAYIMG, grayImg);
Mat histImg = drawHist(grayImg);
imshow(WINDOW_HIST, histImg);
setMouseCallback(WINDOW_HIST, MouseHandle, (void *)&histImg);//(void *)&srcImg传递给void *param
waitKey();
return ;
}
Mat drawHist(Mat &g_srcImg) {
int nRows = ,nCols=;
Mat g_dstImg(nRows,nCols, CV_8UC1, Scalar::all()); // 新建画布
int grayLevel[] = {};
for (int i = ; i < g_srcImg.rows; ++i) {
for (int j = ; j < g_srcImg.cols; ++j) {
grayLevel[(int)g_srcImg.at<uchar>(i, j)]++;
}
}
int MaxCount = arrayMax(grayLevel,);//寻找在处于某一灰度级中个数最多的像素个数
yscaleRate = double(nRows)/MaxCount ;//y缩放比例
double xscaleRate = nCols / ;//x缩放比例
int yAxis[], xAxis[];
for (int m = ; m < ; m++) {
yAxis[m] = int(grayLevel[m]*yscaleRate);
}
//绘制直方图
for (int n = ; n < ; n++) {
if (n == ) {
xAxis[n + ] = xAxis[n];
}
rectangle(g_dstImg, Point(xAxis[n], yAxis[n]), Point( xAxis[n+], ), Scalar(), -);
//-1表示填充矩形框;正值表示不填充矩形框,更方便观察灰度级像素个数的分布;
//但因为后面需要用到填充的情况,故设置成填充状态 } return g_dstImg;
}
//num:数组元素个数;g_max:返回最大值
int arrayMax(int g_arr[],int num) {
int g_max = ;
int i = ;
while (i < num) {
if ( g_arr[i]>= g_max) {
g_max = g_arr[i];
i++;
}
else {
i++;
}
} return g_max;
}
//event 鼠标事件(如按下鼠标左键、左键抬起、鼠标移动等) x、y 鼠标坐标
void MouseHandle(int event, int x, int y, int flags, void *param) {
Mat &g_srcImg = *(Mat *)(param);
Mat g_tempImg = g_srcImg.clone();
int nCount=;//该列白色像素点个数
int nLevelCount = ;
char text[];//存储文本信息
float g_rate;//该列像素点占总像素点个数的比例
imshow(WINDOW_HIST, g_srcImg);
switch (event) {
case EVENT_LBUTTONDOWN:
clickPoint.x = x;
clickPoint.y = y;
downFlag = true;
break;
default:
break;
}
if (downFlag) {
//displayPoint.x = clickPoint.x;
for (int i = ; i < g_srcImg.rows; ++i) {
if (int(g_srcImg.at<uchar>(i, x)) == ) {
++nCount;
}
}
nLevelCount = (nCount / yscaleRate);
g_rate = float(nCount/ yscaleRate) / (srcImg.rows*srcImg.cols);//此处计算的比例不是精确的
cout << "该灰度级像素点个数: " << nLevelCount << ";占总像素个数的比例: "<<g_rate << endl;
sprintf_s(text, "Rate:%f",g_rate );//该灰度级像素个数占总个数的比例;
putText(g_tempImg, text, clickPoint,FONT_HERSHEY_PLAIN,,Scalar(,,));
imshow(WINDOW_HIST, g_tempImg);
downFlag = false;
}
}
4. 结果
结果如下图所示,点击右侧每一级灰度直方图,在控制台中会输出该灰度级像素个数,并显示占总像素比例。

【图像处理】利用C++编写函数,绘制灰度图像直方图的更多相关文章
- 使用matplotlib中的bar函数绘制柱状图
使用柱状图显示三日电影的票房信息 要显示的数据为2018年12月7日-9日四场电影的票房信息 四场电影分别为:无名之辈,狗十三,毒液:知名守卫者,憨豆特工3 2018年12月7日四场电影票房分别为:[ ...
- C++:利用如下公式,编写函数计算∏的值,直到最后一项的绝对值小于e,主程序接收从键盘输入的e,输出∏的值(保留5位小数)。 ∏/4 = 1-1/3+1/5-1/7...
利用如下公式,编写函数计算∏的值,直到最后一项的绝对值小于e,主程序接收从键盘输入的e,输出∏的值(保留5位小数). ∏/4 = 1-1/3+1/5-1/7... #include <iostr ...
- Python 图像处理 OpenCV (16):图像直方图
前文传送门: 「Python 图像处理 OpenCV (1):入门」 「Python 图像处理 OpenCV (2):像素处理与 Numpy 操作以及 Matplotlib 显示图像」 「Python ...
- paper 2:图像处理常用的Matlab函数汇总
一 图像的读写 1 imread imread函数用于读入各种图像文件,如:a=imread('e:\w01.tif') 注:计算机E盘上要有w01相应的.tif文件. 2 imwrite imwri ...
- 利用java编写的盲注脚本
之前在网上见到一个盲注的题目,正好闲来无事,便用java写了个盲注脚本,并记录下过程中的坑 题目源码: <?php header("Content-Type: text/html;ch ...
- 品味性能之道<九>:利用Loadrunner编写socket性能测试脚本简述
一.概述 Loadrunner拥有极为丰富的工具箱,供予我们制造出各种奇妙魔法的能力.其中就有此次要讨论的socket套接字操作. 二.socket概述 ...
- 向日葵远程RCE漏洞分析及漏洞利用脚本编写
0x00 漏洞概述 向日葵是一款免费的,集远程控制电脑.手机.远程桌面连接.远程开机.远程管理.支持内网穿透等功能的一体化远程控制管理软件.如果想要手机远控电脑,或者电脑远控手机可以利用向日葵:如果是 ...
- 利用c++编写bp神经网络实现手写数字识别详解
利用c++编写bp神经网络实现手写数字识别 写在前面 从大一入学开始,本菜菜就一直想学习一下神经网络算法,但由于时间和资源所限,一直未展开比较透彻的学习.大二下人工智能课的修习,给了我一个学习的契机. ...
- iOS基本动画/关键帧动画/利用缓动函数实现物理动画效果
先说下基本动画部分 基本动画部分比较简单, 但能实现的动画效果也很局限 使用方法大致为: #1. 创建原始UI或者画面 #2. 创建CABasicAnimation实例, 并设置keypart/dur ...
随机推荐
- Vector shrink 请求容器降低其容量和size匹配 shrink_to_fit();
一.先从size 和capacity 说起 resize(),设置大小(size); reserve(),设置容量(capacity); size()是分配容器的内存大小,而capacity()只是设 ...
- 【Java8新特性】你知道Java8为什么要引入Lambda表达式吗?
写在前面 这是一道真实的面试题,一个读者朋友出去面试,面试官竟然问他这样一个问题:你说说Java8中为什么引入Lambda表达式?引入Lambda表达式后有哪些好处呢?还好这个朋友对Java8早有准备 ...
- python基础的一些题目
第一部分: 第二部分: 第三部分:
- D. Misha, Grisha and Underground 树链剖分
D. Misha, Grisha and Underground 这个题目算一个树链剖分的裸题,但是这个时间复杂度注意优化. 这个题目可以选择树剖+线段树,时间复杂度有点高,比较这个本身就有n*log ...
- Leetcode_236. 二叉树的最近公共祖先
求二叉树的LCA code /** * Definition for a binary tree node. * struct TreeNode { * int val; * TreeNode *le ...
- 一分钟明白MySQL聚簇索引和非聚簇索引
MySQL的InnoDB索引数据结构是B+树,主键索引叶子节点的值存储的就是MySQL的数据行,普通索引的叶子节点的值存储的是主键值,这是了解聚簇索引和非聚簇索引的前提 什么是聚簇索引? 很简单记住一 ...
- 一步步打造QQ群发消息群发器
最近为了做公众号号推广,吸粉,然后加了几百个QQ群,感觉QQ群的群发效果还是不错的,一天能捞到100个粉丝左右,好的时候也有200个,少的时候几十个,但是由于太多的群了,手工一个个点击开来群发,几百个 ...
- Android 8.1 关机充电动画(一)模式选择
system:Android 8.1 platform:RK3326/PX30 uboot kernel Android 8.1 关机充电动画(一)模式选择 Android 8.1 关机充电动画(二) ...
- 3-JVM垃圾回收算法和垃圾收集器
垃圾回收算法和垃圾收集器 1.什么是垃圾回收 对于内存当中无用的对象进行回收,如何去判断一个对象是不是无用的对象. 引用计数法: 每个对象中都会存储一个引用计数,每增加一个引用就+1,消失一个引用就- ...
- python语法学习第十天--魔法方法
魔法方法二!!! 属性访问:在对属性任何操作时,都会调用 有关属性 __getattr__(self, name) 定义当用户试图获取一个不存在的属性时的行为 __getattribute__(s ...