首页 > 其他 > 详细

3.2直方图处理

时间:2018-08-13 23:12:57      阅读:274      评论:0      收藏:0      [点我收藏+]

3.2.1灰度直方图

技术分享图片
 1 //////3.2.1灰度直方图方法2:
 2 ////Source Code:https://blog.csdn.net/gone_huilin/article/details/53222806
 3 #include <opencv2\opencv.hpp>  
 4 int main()
 5 {
 6     // 图像源获取及判断 
 7     cv::Mat Image, ImageGray;
 8     Image = cv::imread("D:\\lena.jpg");
 9     if (Image.empty())
10         return -1;
11     cv::imshow("Image", Image);
12     // 转换为灰度图像
13     cv::cvtColor(Image, ImageGray, CV_BGR2GRAY);
14     // 定义直方图参数
15     const int channels[1] = { 0 };
16     const int histSize[1] = { 256 };
17     float pranges[2] = { 0,255 };
18     const float* ranges[1] = { pranges };
19     cv::MatND hist;
20     // 计算直方图
21     cv::calcHist(&ImageGray, 1, channels, cv::Mat(), hist, 1,
22         histSize, ranges);
23     // 初始化画布参数
24     int hist_w = 500;
25     int hist_h = 500;
26     int nHistSize = 255;
27     // 区间
28     int bin_w = cvRound((double)hist_w / nHistSize);
29     cv::Mat histImage(hist_w, hist_h,
30         CV_8UC3, cv::Scalar(0, 0, 0));
31     // 将直方图归一化到范围 [ 0, histImage.rows ]
32     normalize(hist, hist, 0, histImage.rows,
33         cv::NORM_MINMAX, -1, cv::Mat());
34     // 在直方图画布上画出直方图
35     for (int i = 1; i < nHistSize; i++)
36     {
37         line(histImage, cv::Point(bin_w*(i - 1),
38             hist_h - cvRound(hist.at<float>(i - 1))),
39             cv::Point(bin_w*(i),
40                 hist_h - cvRound(hist.at<float>(i))),
41             cv::Scalar(0, 0, 255), 2, 8, 0);
42     }
43     // 显示直方图
44     cv::imshow("histImage", histImage);
45     cv::waitKey();
46     return 0;
47 }
View Code

技术分享图片

技术分享图片

 

 

技术分享图片
 1 ////3.2.1灰度直方图
 2 ////Source Code:https://blog.csdn.net/qq_20823641/article/details/51932798
 3 #include "opencv2/highgui/highgui.hpp"
 4 #include "opencv2/imgproc/imgproc.hpp"
 5 #include <iostream>
 6 using namespace cv;
 7 using namespace std;
 8 
 9 void Help()
10 {
11     printf("\n\n\t\t\t欢迎来到直方图的世界!\n");
12     printf("\n\n  ----------------------------------------------------------------------------\n");
13 }
14 
15 
16 int main()
17 {
18     Mat srcImage = imread("D:\\lena.jpg");
19     Mat grayImage;
20     if (!srcImage.data) 
21     { 
22         cout << "fail to load image" << endl;     
23         return 0; 
24     }
25     imshow("原图", srcImage);
26     cvtColor(srcImage, grayImage, CV_RGB2GRAY);
27     imshow("灰度图", grayImage);
28 
29     system("color 1F");
30     Help();
31 
32     MatND dstHist;       // 在cv中用CvHistogram *hist = cvCreateHist
33     int dims = 1;
34     float hranges[2] = { 0, 255 };
35     const float *ranges[1] = { hranges };   // 这里需要为const类型
36     int size = 256;
37     int channels = 0;
38     //计算图像的直方图
39     calcHist(&grayImage, 1, &channels, Mat(), dstHist, dims, &size, ranges);    // cv 中是cvCalcHist
40     int scale = 1;
41     Mat dstImage(size * scale, size, CV_8U, Scalar(0));
42     //获取最大值和最小值
43     double minValue = 0;
44     double maxValue = 0;
45     minMaxLoc(dstHist, &minValue, &maxValue, 0, 0);  //  在cv中用的是cvGetMinMaxHistValue
46                                                      //绘制出直方图
47     int hpt = saturate_cast<int>(0.9 * size);
48     for (int i = 0; i < 256; i++)
49     {
50         float binValue = dstHist.at<float>(i);           //   注意hist中是float类型    
51         int realValue = saturate_cast<int>(binValue * hpt / maxValue);
52         //rectangle(dstImage,Point(i*scale, size - 1), Point((i+1)*scale - 1, size - realValue), Scalar(255));
53         line(dstImage, Point(i*scale, size - 1), Point((i + 1)*scale - 1, size - realValue), Scalar(255));
54     }
55     imshow("一维直方图", dstImage);
56     waitKey(0);
57     return 0;
58 }
View Code

技术分享图片

技术分享图片技术分享图片

技术分享图片

 

 

 

技术分享图片
////Source Code:https://blog.csdn.net/qq_20823641/article/details/51932798
#include "cv.h"
#include "highgui.h"
#include <stdio.h>
#include <ctype.h>

#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include <iostream>

using namespace std;
using namespace cv;


IplImage *src = 0;
IplImage *histimg = 0;
CvHistogram *hist = 0;

int hdims = 50;     // 划分HIST的初始个数,越高越精确

                    //滚动条函数
void HIST(int t)
{
    float hranges_arr[] = { 0,255 };
    float* hranges = hranges_arr;
    int bin_w;
    int bin_u;
    float max;
    int i;
    char string[10];
    CvFont font;
    cvInitFont(&font, CV_FONT_HERSHEY_PLAIN, 1, 1, 0, 1, 8);//字体结构初始化
    if (hdims == 0)
    {
        printf("直方图条数不能为零!\n");
    }
    else
    {
        hist = cvCreateHist(1, &hdims, CV_HIST_ARRAY, &hranges, 1);  // 创建直方图
        histimg = cvCreateImage(cvSize(800, 512), 8, 3);
        cvZero(histimg);
        cvCalcHist(&src, hist, 0, 0); // 计算直方图
        cvGetMinMaxHistValue(hist, NULL, &max, NULL, NULL);//寻找最大值及其位置
                                                           //printf("max_val:%f \n",max_val);
        cvZero(histimg);

        double bin_w = (double)histimg->width / hdims;  // hdims: 条的个数,则 bin_w 为条的宽度
        double bin_u = (double)histimg->height / max;  //// max: 最高条的像素个数,则 bin_u 为单个像素的高度

                                                       // 画直方图
        for (int i = 0; i<hdims; i++)
        {
            CvPoint p0 = cvPoint(i*bin_w, histimg->height);
            int val = cvGetReal1D(hist->bins, i);
            CvPoint p1 = cvPoint((i + 1)*bin_w, histimg->height - cvGetReal1D(hist->bins, i)*bin_u);
            cvRectangle(histimg, p0, p1, cvScalar(0, 255), 1, 8, 0);
        }
        //画纵坐标刻度(像素个数)
        int kedu = 0;
        for (int i = 1; kedu<max; i++)
        {
            kedu = i*max / 10;
            _itoa_s(kedu, string, 10);//把一个整数转换为字符串
                                   //在图像中显示文本字符串
            cvPutText(histimg, string, cvPoint(0, histimg->height - kedu*bin_u), &font, CV_RGB(0, 255, 255));
        }
        //画横坐标刻度(像素灰度值)
        kedu = 0;
        for (int i = 1; kedu<256; i++)
        {
            kedu = i * 20;
            _itoa_s(kedu, string, 10);//把一个整数转换为字符串
                                   //在图像中显示文本字符串
            cvPutText(histimg, string, cvPoint(kedu*(histimg->width / 256), histimg->height), &font, CV_RGB(255, 0, 0));
        }

        cvShowImage("Histogram", histimg);
    }
}

int main(int argc, char** argv)
{
    argc = 2;
    argv[1] = "D:\\lena.jpg";

    if (argc != 2 || (src = cvLoadImage(argv[1], 0)) == NULL)  // force to gray image
        return -1;

    cvNamedWindow("src", 1);
    cvShowImage("src", src);
    cvNamedWindow("Histogram", 1);

    cvCreateTrackbar("hdims", "src", &hdims, 256, HIST);
    HIST(0);
    cvWaitKey(0);

    cvDestroyWindow("src");
    cvDestroyWindow("Histogram");
    cvReleaseImage(&src);
    cvReleaseImage(&histimg);
    cvReleaseHist(&hist);

    return 0;
}
View Code

技术分享图片

技术分享图片

 

 参考:

https://blog.csdn.net/xiaowei_cqu/article/details/7600666

https://www.cnblogs.com/wangguchangqing/p/7098213.html 

 

3.2.2 H-S直方图

 

技术分享图片
 1 ////Source Code: https://blog.csdn.net/uestc_c2_403/article/details/72814455
 2 #include "opencv2/highgui/highgui.hpp"
 3 #include "opencv2/imgproc/imgproc.hpp"
 4 #include "opencv2/opencv.hpp"
 5 #include "opencv2/core/core.hpp"
 6 #include <stdio.h>
 7 #include <string>
 8 using namespace std;
 9 using namespace cv;
10 int main()
11 {
12     cv::Mat hsvMat;
13     cv::Mat srcImage = cv::imread("D:\\lena.jpg");
14 
15     if (srcImage.empty())
16     {
17         return -1;
18     }
19     cv::imshow("原图像", srcImage);
20     //灰度转换
21     cv::Mat srcGray;
22     cv::cvtColor(srcImage, hsvMat, CV_BGR2HSV);
23     cv::imshow("hsvMat", hsvMat);
24     //初始化灰度阶参数
25     int hbins = 30;
26     int sbins = 32;
27     int histSize[] = { hbins, sbins };
28     //灰度变化范围设置
29     float hranges[] = { 0, 180 };
30     //饱和度变化范围
31     float sranges[] = { 0, 256 };
32     const float* ranges[] = { hranges, sranges };
33     cv::MatND hist;
34 
35     //选取计算直方图通道
36     int channels[] = { 0, 1 };
37     //计算当前通道直方图
38     cv::calcHist(&hsvMat, 1, channels, cv::Mat(), hist, 2, histSize, ranges, true, false);
39     double maxVal = 0;
40     //找到直方图最大值
41     cv::minMaxLoc(hist, 0, &maxVal, 0, 0);
42     int scale = 10;
43     cv::Mat histImage = cv::Mat::zeros(sbins*scale, hbins * scale, CV_8UC3);
44     //遍历H、S通道
45     for (int h = 0; h < hbins; h++)
46     {
47         for (int s = 0; s < sbins; s++)
48         {
49             float binVal = hist.at<float>(h, s);
50             //根据最大值计算变化范围
51             int intensity = cvRound(binVal * 255 / maxVal);
52             //绘图显示
53             cv::rectangle(histImage, cv::Point(h*scale, s*scale),
54                 cv::Point((h + 1)*scale - 1, (s + 1)*scale - 1), cv::Scalar::all(intensity), CV_FILLED);
55         }
56     }
57 
58     cv::imshow("H-S Histogram", histImage);
59     cv::waitKey(0);
60     return 0;
61 }
View Code

技术分享图片技术分享图片技术分享图片

 

3.2.3B-G-R直方图 

技术分享图片
 1 ////https://blog.csdn.net/gone_huilin/article/details/53222853
 2 #include <opencv2/opencv.hpp>
 3 #include <vector>
 4 #include <iostream>
 5 
 6 using namespace cv;
 7 
 8 int main()
 9 {
10     cv::Mat srcImage = cv::imread("D:\\lena.jpg");
11     if (!srcImage.data)
12         return 1;
13     cv::imshow("srcImage", srcImage);
14     cv::Mat bgr_planes[3];
15     cv::split(srcImage, bgr_planes);
16 
17     // 初始化直方图计算参数
18     int histSize = 256;
19     float range[] = { 0, 256 };
20     const float* histRange = { range };
21     bool uniform = true;
22     bool accumulate = false;
23     cv::Mat b_hist, g_hist, r_hist;
24     // 计算各个通道的直方图
25     calcHist(&bgr_planes[0], 1, 0, cv::Mat(), b_hist, 1,
26         &histSize, &histRange, uniform, accumulate);
27     calcHist(&bgr_planes[1], 1, 0, cv::Mat(), g_hist, 1,
28         &histSize, &histRange, uniform, accumulate);
29     calcHist(&bgr_planes[2], 1, 0, cv::Mat(), r_hist, 1,
30         &histSize, &histRange, uniform, accumulate);
31     // 设置直方图绘图参数
32     int hist_w = 640; int hist_h = 512;
33     int bin_w = cvRound((double)hist_w / histSize);
34     cv::Mat histImage(hist_h, hist_w,
35         CV_8UC3, cv::Scalar(0, 0, 0));
36     // 分别归一化直方图到[ 0, histImage.rows ]
37     normalize(b_hist, b_hist, 0, histImage.rows, NORM_MINMAX,
38         -1, Mat());
39     normalize(g_hist, g_hist, 0, histImage.rows, NORM_MINMAX,
40         -1, Mat());
41     normalize(r_hist, r_hist, 0, histImage.rows, NORM_MINMAX,
42         -1, Mat());
43     // 分别对每个通道进行绘图
44     for (int i = 1; i < histSize; i++)
45     {
46         line(histImage, Point(bin_w*(i - 1),
47             hist_h - cvRound(b_hist.at<float>(i - 1))),
48             Point(bin_w*(i), hist_h -
49                 cvRound(b_hist.at<float>(i))),
50             Scalar(255, 0, 0), 2, 8, 0);
51         line(histImage, Point(bin_w*(i - 1), hist_h -
52             cvRound(g_hist.at<float>(i - 1))),
53             Point(bin_w*(i), hist_h -
54                 cvRound(g_hist.at<float>(i))),
55             Scalar(0, 255, 0), 2, 8, 0);
56         line(histImage, Point(bin_w*(i - 1), hist_h -
57             cvRound(r_hist.at<float>(i - 1))),
58             Point(bin_w*(i), hist_h -
59                 cvRound(r_hist.at<float>(i))),
60             Scalar(0, 0, 255), 2, 8, 0);
61     }
62     imshow("calcHist", histImage);
63     cv::waitKey(0);
64     return 0;
65 }
View Code

技术分享图片技术分享图片

技术分享图片
 1 ////https://blog.csdn.net/dainesao/article/details/79184934
 2 ////3.2.3B-G-R直方图方法2:
 3 #include <opencv2/opencv.hpp>  
 4 #include <opencv2/imgproc/imgproc.hpp>  
 5 
 6 using namespace cv;
 7 
 8 int main()
 9 {
10     Mat srcImage;
11     srcImage = imread("D:\\lena.jpg");
12     imshow("素材图", srcImage);
13     int bins = 256;
14     int hist_size[] = { bins };
15     float range[] = { 0, 256 };
16     const float* ranges[] = { range };
17     MatND redHist, grayHist, blueHist;
18 
19     //进行直方图的计算(红色分量部分)
20     int channels_r[] = { 0 };
21     calcHist(&srcImage, 1, channels_r, Mat(),
22         redHist, 1, hist_size, ranges,
23         true, false);
24 
25     //进行直方图的计算(绿色分量部分)
26     int channels_g[] = { 1 };
27     calcHist(&srcImage, 1, channels_g, Mat(), // do not use mask
28         grayHist, 1, hist_size, ranges,
29         true, // the histogram is uniform
30         false);
31 
32     //进行直方图的计算(蓝色分量部分)
33     int channels_b[] = { 2 };
34     calcHist(&srcImage, 1, channels_b, Mat(), // do not use mask
35         blueHist, 1, hist_size, ranges,
36         true, // the histogram is uniform
37         false);
38 
39     //-----------------------绘制出三色直方图------------------------
40     //参数准备
41     double maxValue_red, maxValue_green, maxValue_blue;
42     minMaxLoc(redHist, 0, &maxValue_red, 0, 0);
43     minMaxLoc(grayHist, 0, &maxValue_green, 0, 0);
44     minMaxLoc(blueHist, 0, &maxValue_blue, 0, 0);
45     int scale = 1;
46     int histHeight = 256;
47     Mat histImage = Mat::zeros(histHeight, bins * 3, CV_8UC3);
48 
49     //正式开始绘制
50     for (int i = 0; i<bins; i++)
51     {
52         //参数准备
53         float binValue_red = redHist.at<float>(i);
54         float binValue_green = grayHist.at<float>(i);
55         float binValue_blue = blueHist.at<float>(i);
56         int intensity_red = cvRound(binValue_red*histHeight / maxValue_red);  //要绘制的高度
57         int intensity_green = cvRound(binValue_green*histHeight / maxValue_green);  //要绘制的高度
58         int intensity_blue = cvRound(binValue_blue*histHeight / maxValue_blue);  //要绘制的高度
59 
60                                                                                  //绘制红色分量的直方图
61         rectangle(histImage, Point(i*scale, histHeight - 1),
62             Point((i + 1)*scale - 1, histHeight - intensity_red),
63             Scalar(255, 0, 0));
64 
65         //绘制绿色分量的直方图
66         rectangle(histImage, Point((i + bins)*scale, histHeight - 1),
67             Point((i + bins + 1)*scale - 1, histHeight - intensity_green),
68             Scalar(0, 255, 0));
69 
70         //绘制蓝色分量的直方图
71         rectangle(histImage, Point((i + bins * 2)*scale, histHeight - 1),
72             Point((i + bins * 2 + 1)*scale - 1, histHeight - intensity_blue),
73             Scalar(0, 0, 255));
74 
75     }
76 
77     //在窗口中显示出绘制好的直方图
78     imshow("图像的RGB直方图", histImage);
79     waitKey(0);
80     return 0;
81 }
View Code

 技术分享图片技术分享图片

 

 3.2.4 自定义直方图

技术分享图片
 1 ////https://blog.csdn.net/gone_huilin/article/details/53222859
 2 #include <opencv2/opencv.hpp>
 3 int main()
 4 {
 5     // 图像获取及判断
 6     cv::Mat srcImage = cv::imread("D:\\lena.jpg");
 7     if (!srcImage.data)
 8         return 1;
 9     cv::imshow("srcImage", srcImage);
10     // 灰度转换  
11     cv::Mat srcGray;
12     cv::cvtColor(srcImage, srcGray, CV_BGR2GRAY);
13     // 初始化直方图计算参数
14     const int channels[1] = { 0 };
15     const int histSize[1] = { 256 };
16     // 设定区间[0 60],[61 120],[121 160],[161 220],[221 255]
17     float hranges[6] = { 0, 60, 120, 160, 220, 255 };
18     const float* ranges[1] = { hranges };
19     cv::MatND hist;
20     // 计算直方图
21     cv::calcHist(&srcGray, 1,
22         channels, cv::Mat(),
23         hist, 1, histSize,
24         ranges);
25     // 求直方图中最大值
26     double maxHist = 0;
27     cv::minMaxLoc(hist, 0, &maxHist, 0, 0);
28     // 设置直方图绘图参数
29     int hist_Size = hist.rows;
30     cv::Mat histImg(hist_Size, hist_Size,
31         CV_8U, cv::Scalar(255));
32     // 直方图绘制
33     for (int h = 0; h < hist_Size; h++)
34     {
35         float binVal = hist.at<float>(h);
36         //归一化 根据最大值计算变换范围 
37         int intensity = static_cast<int>(binVal *
38             hist_Size / maxHist);
39         // 绘图直方图信息
40         cv::line(histImg, cv::Point(h, hist_Size),
41             cv::Point(h, hist_Size - intensity),
42             cv::Scalar::all(0));
43     }
44     cv::imshow("histImg", histImg);
45     cv::waitKey(0);
46     return 0;
47 }
View Code

技术分享图片技术分享图片

3.2.5灰度直方图均衡

技术分享图片
 1 //////////////////////////3.2.5灰度直方图均衡///////////////////////////
 2 ////https://blog.csdn.net/linqianbi/article/details/78603406
 3 #include<opencv2\opencv.hpp>
 4 #include<cmath>
 5 #include<iostream>
 6 using namespace cv;
 7 using namespace std;
 8 Mat MyequalizeHist(Mat &srcImage)
 9 {
10     int nRows = srcImage.rows;
11     int nCols = srcImage.cols;
12 
13     int nSumPix[256];
14     double nProDis[256];
15     double nSumProDis[256];
16     int EqualizeSumPix[256];
17 
18     for (int i = 0; i < 256; i++)
19     {
20         nSumPix[i] = 0;
21         nProDis[i] = 0.0;
22         nSumProDis[i] = 0.0;
23         EqualizeSumPix[i] = 0;
24     }
25 
26     for (int i = 0; i < nRows; i++)
27     {
28         for (int j = 0; j < nCols; j++)
29         {
30             nSumPix[(int)srcImage.at<uchar>(i, j)]++;
31         }
32     }
33 
34 
35     for (int i = 0; i < 256; i++)
36     {
37         nProDis[i] = (double)nSumPix[i] / (nRows * nCols);
38     }
39 
40 
41     nSumProDis[0] = nProDis[0];
42 
43 
44     for (int i = 1; i < 256; i++)
45     {
46         nSumProDis[i] = nSumProDis[i - 1] + nProDis[i];
47     }
48 
49 
50     for (int i = 0; i < 256; i++)
51     {
52         EqualizeSumPix[i] = cvRound((double)nSumProDis[i] * 255);
53     }
54 
55     Mat resultImage(nRows, nCols, srcImage.type());
56     for (int i = 0; i < nRows; i++)
57     {
58 
59         for (int j = 0; j < nCols; j++)
60         {
61 
62             resultImage.at<uchar>(i, j) = EqualizeSumPix[(int)srcImage.at<uchar>(i, j)];
63         }
64     }
65     return resultImage;
66 
67 }
68 int main()
69 {
70     Mat srcIamge = imread("D:\\lena.jpg");
71     if (!srcIamge.data)
72     {
73         printf("image could not load...\n");
74         return -1;
75     }
76     Mat srcGray;
77     //转化为灰度图并且显示
78     cvtColor(srcIamge, srcGray, CV_BGR2GRAY);
79     imshow("srcGray", srcGray);
80 
81     Mat resultImage = MyequalizeHist(srcGray);
82     imshow("res", resultImage);
83 
84     waitKey(0);
85     return 0;
86 }
View Code

 技术分享图片技术分享图片

参考:

https://blog.csdn.net/linqianbi/article/details/78603406

https://blog.csdn.net/xiaowei_cqu/article/details/7606607

https://www.cnblogs.com/wangguchangqing/p/7098213.html

https://blog.csdn.net/sunmc1204953974/article/details/50606395

https://blog.csdn.net/zhulf0804/article/details/52770613/ 

 

3.2直方图处理

原文:https://www.cnblogs.com/thebreakofdawn/p/9471537.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!