python 直方图匹配_基于OpenCV的直方图匹配

def generate_histogram(img, do_print):"""@params: img: can be a grayscale or color image. We calculate the Normalized histogram of this image.@params: do_print: if or not print the result histogram@return: will return both histogram and the grayscale image"""if len(img.shape) == 3: # img is colorful, so we convert it to grayscalegr_img = np.mean(img, axis=-1)else:gr_img = img'''now we calc grayscale histogram'''gr_hist = np.zeros([256])

for x_pixel in range(gr_img.shape[0]):for y_pixel in range(gr_img.shape[1]):pixel_value = int(gr_img[x_pixel, y_pixel])gr_hist[pixel_value] += 1

'''normalizing the Histogram'''gr_hist /= (gr_img.shape[0] * gr_img.shape[1])if do_print:print_histogram(gr_hist, name="n_h_img", title="Normalized Histogram")return gr_hist, gr_img

def print_histogram(_histrogram, name, title):plt.figureplt.title(title)plt.plot(_histrogram, color='#ef476f')plt.bar(np.arange(len(_histrogram)), _histrogram, color='#b7b7a4')plt.ylabel('Number of Pixels')plt.xlabel('Pixel Value')plt.savefig("hist_" + name)

代码1:生成直方图

在大多数情况下,当我们创建直方图时,我们通过将每个强度值的像素数除以归一化因子(即图像宽度和图像高度的乘积)来对直方图进行归一化。为了便于使用,如果generate_histogram函数的输入图像是彩色图像,我们首先将其转换为灰度图像(请参见第6行)。

03. 如何均衡图像直方图?

直方图均衡化通常用于增强图像的对比度。因此,该技术不能保证始终提高图像质量。计算CDF(累积分布函数)是均衡图像直方图的常用方法。在图2中,我们计算了在图1中创建的样本图像的CDF。此外,在图3中,我们显示了先前样本的均衡直方图。

ba30983da250a2ae84690c9d9d988a4a.png

图2:计算CDF。

c10b0c446edb5936dea0d0a923243410.png

图3:均方图。

为了计算python中的均衡直方图,我们创建了以下代码:

def equalize_histogram(img, histo, L):eq_histo = np.zeros_like(histo)en_img = np.zeros_like(img)for i in range(len(histo)):eq_histo[i] = int((L - 1) * np.sum(histo[0:i]))print_histogram(eq_histo, name="eq_"+str(index), title="Equalized Histogram")'''enhance image as well:'''for x_pixel in range(img.shape[0]):for y_pixel in range(img.shape[1]):pixel_val = int(img[x_pixel, y_pixel])en_img[x_pixel, y_pixel] = eq_histo[pixel_val]'''creating new histogram'''hist_img, _ = generate_histogram(en_img, print=False, index=index)print_img(img=en_img, histo_new=hist_img, histo_old=histo, index=str(index), L=L)return eq_histo

代码2:均衡直方图

这是我们拍摄的3张不同图片,并用作示例。如图4所示,对于第一个图像,直方图显示低强度像素的数量多于明亮像素。对于第二张图像,情况完全相反,其中较亮像素的密度远大于较暗像素的密度。第三张图片似乎具有半正态直方图。

768324f0c375b6d2c074cd190a78068e.png

ce70d75488dc32bfae023a58587a821b.png

834175009d707c0655c750d7c17fc45f.png

图4:三种不同类型的图像及其直方图和均等的直方图。

使用均衡直方图增强图像

如前所述,我们可以使用图像的均衡直方图修改图像的对比度。如代码2第12行所示,对于输入图像中的每个像素,我们可以使用其均等值。结果可能比原始图像更好,但不能保证。在图5中,我们描述了3张图像的修改版本。如图所示,使用其均等的直方图修改图像会产生对比度更高的图像。此功能在许多计算机视觉任务中很有用。

b67095c1b65b9b72b5d9e922f9d94bd4.png

b3c90e33756c28d3526b0f863f31f1c2.png

cf1faef13c98e7f4b32bd2dc819f6677.png

图5:使用均衡直方图的对比度修改。最左列是原始图像。中间一栏是对比度修改的结果。

最右边的列是修改后的图像的直方图。

04. 什么是直方图匹配?

假设我们有两个图像,每个图像都有其特定的直方图。因此,我们想在进一步解决此问题之前,是否可以根据另一幅图像的对比度来修改一幅图像?答案是肯定的。实际上,这就是直方图匹配的定义。换句话说,给定图像A和B,可以根据B修改A的对比度。

当我们要统一一组图像的对比度时,直方图匹配非常有用。实际上,直方图均衡也可以视为直方图匹配,因为我们将输入图像的直方图修改为与正态分布相似。

为了匹配图像A和B的直方图,我们需要首先均衡两个图像的直方图。然后,我们需要使用均衡后的直方图将A的每个像素映射到B。然后,我们基于B修改A的每个像素。

让我们使用图6中的以下示例来阐明以上段落。

0486b502adfd10091f5fbf91dea902ad.png

图6:直方图匹配

在图6中,我们将图像A作为输入图像,将图像B作为目标图像。我们要基于B的分布来修改A的直方图。第一步,我们计算A和B的直方图和均等直方图。然后,我们需要根据该值映射A的每个像素它的均衡直方图求B的值。因此,例如,对于A中强度级别为 0的 像素,A均衡直方图的对应值为 4。 现在,我们看一下B均衡直方图并找到强度值对应于4,即0。因此我们将 0 强度从A映射到 0 从B开始。对于A的所有强度值,我们继续进行。如果从A到B的均衡直方图中没有映射,我们只需要选择最接近的值即可。

def find_value_target(val, target_arr):key = np.where(target_arr == val)[0]if len(key) == 0:key = find_value_target(val+1, target_arr)if len(key) == 0:key = find_value_target(val-1, target_arr)vvv = key[0]return vvv

def match_histogram(inp_img, hist_input, e_hist_input, e_hist_target, _print=True):'''map from e_inp_hist to 'target_hist '''en_img = np.zeros_like(inp_img)tran_hist = np.zeros_like(e_hist_input)for i in range(len(e_hist_input)):tran_hist[i] = find_value_target(val=e_hist_input[i], target_arr=e_hist_target)print_histogram(tran_hist, name="trans_hist_", title="Transferred Histogram")'''enhance image as well:'''for x_pixel in range(inp_img.shape[0]):for y_pixel in range(inp_img.shape[1]):pixel_val = int(inp_img[x_pixel, y_pixel])en_img[x_pixel, y_pixel] = tran_hist[pixel_val]'''creating new histogram'''hist_img, _ = generate_histogram(en_img, print=False, index=3)print_img(img=en_img, histo_new=hist_img, histo_old=hist_input, index=str(3), L=L)

代码3:Python中的直方图匹配

b2dd278111d33fcddd3b8c8630551e87.png

图7:直方图匹配示例。我们修改了左图像的直方图以匹配中心图像的直方图。

图7示出了直方图匹配的示例。如大家所见,尽管最左边的图像是明亮的图像,但就对比度级别而言,可以将中心图像视为更好的图像。因此,我们决定使用中心图像的收缩来修改最左边的图像。结果,即最右边的图像已得到改善。

代码链接:https://github.com/aliprf/CV-HistogramMatching返回搜狐,查看更多

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是基于OpenCV实现直方图匹配算法的代码示例。该示例使用C++编写,在Visual Studio下运行。 首先,需要包含OpenCV库: ```cpp #include <opencv2/opencv.hpp> ``` 然后,定义直方图匹配函数: ```cpp cv::Mat histMatch(cv::Mat& src, cv::Mat& dst) { cv::Mat srcHist, dstHist; cv::Mat srcCumHist, dstCumHist; // 计算源图像和目标图像的直方图 cv::calcHist(&src, 1, nullptr, cv::Mat(), srcHist, 1, &histSize, &histRange, true, false); cv::calcHist(&dst, 1, nullptr, cv::Mat(), dstHist, 1, &histSize, &histRange, true, false); // 计算源图像和目标图像的累计直方图 cv::normalize(srcHist, srcCumHist, 0, 255, cv::NORM_MINMAX, CV_8UC1); cv::normalize(dstHist, dstCumHist, 0, 255, cv::NORM_MINMAX, CV_8UC1); // 创建查找表 cv::Mat lut(1, 256, CV_8UC1); int s = 0; for (int i = 0; i < 256; i++) { int d = s; while (d < 255 && dstCumHist.at<uchar>(d) < srcCumHist.at<uchar>(i)) { d++; } lut.at<uchar>(i) = d; s = d; } // 应用查找表 cv::Mat result; cv::LUT(src, lut, result); return result; } ``` 该函数接受两个参数:源图像和目标图像。它计算源图像和目标图像的直方图,并使用累计直方图创建一个查找表。最后,它应用该查找表将源图像与目标图像匹配,返回匹配后的图像。 可以将此函数与其他函数结合使用,以创建完整的程序。例如,可以使用以下代码读取图像,调用直方图匹配函数,并显示结果: ```cpp int main() { cv::Mat src = cv::imread("src.jpg", cv::IMREAD_GRAYSCALE); cv::Mat dst = cv::imread("dst.jpg", cv::IMREAD_GRAYSCALE); cv::Mat result = histMatch(src, dst); cv::imshow("Result", result); cv::waitKey(0); return 0; } ``` 在此示例中,读取了两个灰度图像,并将其传递给直方图匹配函数。然后,将匹配后的结果显示在屏幕上。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值