Python OpenCV -- 直方图均衡化(十三)
2014-02-25 01:18
363 查看
直方图均衡化
直方图是图像中像素强度分布的图形表达方式。它统计了每一个强度值所具有的像素个数。
直方图均衡化是通过拉伸像素强度分布范围来增强图像对比度的一种方法。
通过上图可以看到像素主要集中在中间的一些强度值上。直方图均衡化要做的就是 拉伸 这个范围(绿圈圈出来的部分) 少有像素分布
其上的 强度值,对其应用均衡化后,得到中间图所示的直方图。
原理:
均衡化指的是把一个分布(给定的直方图)映射 到另一个分布(一个更宽更统一的强度值分布),所以强度值分布会在整个范围内展开。
要想实现均衡化的效果,映射函数应该是一个 累积分布函数 (cdf),对于直方图
,它的
累积分布
是:
要使用其作为映射函数,我们必须对最大值为255(或者用图像的最大强度值)的累积分布
进行归一化。同上例,累积分布函数为:
最后我们使用一个简单的映射过程来获得均衡化后像素的强度值:
在 Opencv Python 实现
1. 拉伸直方图(使用查询表方法)
先检测图像非0的最低(imin)强度值和最高(强度值)。将最低值 imin 设为0,最高值 imax 设为255.中间值按 255.0 * (i - imin) / (imax - imin) + 0.5)
的形式设置。
示例(这是使用sunny2038 提供的示例代码):
#!/usr/bin/env python
# encoding: utf-8
import cv2
import numpy as np
image = cv2.imread("113.jpg", 0)
lut = np.zeros(256, dtype = image.dtype )#创建空的查找表
hist= cv2.calcHist([image], #计算图像的直方图
[0], #使用的通道
None, #没有使用mask
[256], #it is a 1D histogram
[0.0,255.0])
minBinNo, maxBinNo = 0, 255
#计算从左起第一个不为0的直方图柱的位置
for binNo, binValue in enumerate(hist):
if binValue != 0:
minBinNo = binNo
break
#计算从右起第一个不为0的直方图柱的位置
for binNo, binValue in enumerate(reversed(hist)):
if binValue != 0:
maxBinNo = 255-binNo
break
print minBinNo, maxBinNo
#生成查找表
for i,v in enumerate(lut):
print i
if i < minBinNo:
lut[i] = 0
elif i > maxBinNo:
lut[i] = 255
else:
lut[i] = int(255.0*(i-minBinNo)/(maxBinNo-minBinNo)+0.5)
#计算,调用OpenCV cv2.LUT函数,参数 image -- 输入图像,lut -- 查找表
result = cv2.LUT(image, lut)
cv2.imshow("Result", result)
cv2.imwrite("LutImage.jpg", result)
cv2.waitKey(0)
cv2.destroyAllWindows()
效果图(左边是原图)
2.Python Numpy直方图均衡化
示例(示例代码使用 sunny2038 博客提供的)
#!/usr/bin/env python
# encoding: utf-8
import cv2
import numpy as np
image = cv2.imread("113.jpg", 0)
lut = np.zeros(256, dtype = image.dtype )#创建空的查找表
hist,bins = np.histogram(image.flatten(),256,[0,256])
cdf = hist.cumsum() #计算累积直方图
cdf_m = np.ma.masked_equal(cdf,0) #除去直方图中的0值
cdf_m = (cdf_m - cdf_m.min())*255/(cdf_m.max()-cdf_m.min())#等同于前面介绍的lut[i] = int(255.0 *p[i])公式
cdf = np.ma.filled(cdf_m,0).astype('uint8') #将掩模处理掉的元素补为0
#计算
result2 = cdf[image]
result = cv2.LUT(image, cdf)
cv2.imshow("OpenCVLUT", result)
cv2.imshow("NumPyLUT", result2)
cv2.waitKey(0)
cv2.destroyAllWindows()
效果图:
本文参考和转载:
http://www.opencv.org.cn/opencvdoc/2.3.2/html/doc/tutorials/imgproc/histograms/histogram_equalization/histogram_equalization.html#histogram-equalization http://blog.csdn.net/sunny2038/article/details/9403059
直方图是图像中像素强度分布的图形表达方式。它统计了每一个强度值所具有的像素个数。
直方图均衡化是通过拉伸像素强度分布范围来增强图像对比度的一种方法。
通过上图可以看到像素主要集中在中间的一些强度值上。直方图均衡化要做的就是 拉伸 这个范围(绿圈圈出来的部分) 少有像素分布
其上的 强度值,对其应用均衡化后,得到中间图所示的直方图。
原理:
均衡化指的是把一个分布(给定的直方图)映射 到另一个分布(一个更宽更统一的强度值分布),所以强度值分布会在整个范围内展开。
要想实现均衡化的效果,映射函数应该是一个 累积分布函数 (cdf),对于直方图
,它的
累积分布
是:
要使用其作为映射函数,我们必须对最大值为255(或者用图像的最大强度值)的累积分布
进行归一化。同上例,累积分布函数为:
最后我们使用一个简单的映射过程来获得均衡化后像素的强度值:
在 Opencv Python 实现
1. 拉伸直方图(使用查询表方法)
先检测图像非0的最低(imin)强度值和最高(强度值)。将最低值 imin 设为0,最高值 imax 设为255.中间值按 255.0 * (i - imin) / (imax - imin) + 0.5)
的形式设置。
示例(这是使用sunny2038 提供的示例代码):
#!/usr/bin/env python
# encoding: utf-8
import cv2
import numpy as np
image = cv2.imread("113.jpg", 0)
lut = np.zeros(256, dtype = image.dtype )#创建空的查找表
hist= cv2.calcHist([image], #计算图像的直方图
[0], #使用的通道
None, #没有使用mask
[256], #it is a 1D histogram
[0.0,255.0])
minBinNo, maxBinNo = 0, 255
#计算从左起第一个不为0的直方图柱的位置
for binNo, binValue in enumerate(hist):
if binValue != 0:
minBinNo = binNo
break
#计算从右起第一个不为0的直方图柱的位置
for binNo, binValue in enumerate(reversed(hist)):
if binValue != 0:
maxBinNo = 255-binNo
break
print minBinNo, maxBinNo
#生成查找表
for i,v in enumerate(lut):
print i
if i < minBinNo:
lut[i] = 0
elif i > maxBinNo:
lut[i] = 255
else:
lut[i] = int(255.0*(i-minBinNo)/(maxBinNo-minBinNo)+0.5)
#计算,调用OpenCV cv2.LUT函数,参数 image -- 输入图像,lut -- 查找表
result = cv2.LUT(image, lut)
cv2.imshow("Result", result)
cv2.imwrite("LutImage.jpg", result)
cv2.waitKey(0)
cv2.destroyAllWindows()
效果图(左边是原图)
2.Python Numpy直方图均衡化
示例(示例代码使用 sunny2038 博客提供的)
#!/usr/bin/env python
# encoding: utf-8
import cv2
import numpy as np
image = cv2.imread("113.jpg", 0)
lut = np.zeros(256, dtype = image.dtype )#创建空的查找表
hist,bins = np.histogram(image.flatten(),256,[0,256])
cdf = hist.cumsum() #计算累积直方图
cdf_m = np.ma.masked_equal(cdf,0) #除去直方图中的0值
cdf_m = (cdf_m - cdf_m.min())*255/(cdf_m.max()-cdf_m.min())#等同于前面介绍的lut[i] = int(255.0 *p[i])公式
cdf = np.ma.filled(cdf_m,0).astype('uint8') #将掩模处理掉的元素补为0
#计算
result2 = cdf[image]
result = cv2.LUT(image, cdf)
cv2.imshow("OpenCVLUT", result)
cv2.imshow("NumPyLUT", result2)
cv2.waitKey(0)
cv2.destroyAllWindows()
效果图:
本文参考和转载:
http://www.opencv.org.cn/opencvdoc/2.3.2/html/doc/tutorials/imgproc/histograms/histogram_equalization/histogram_equalization.html#histogram-equalization http://blog.csdn.net/sunny2038/article/details/9403059
相关文章推荐
- python3.6.3+opencv3.3.0学习笔记十三--基于HOG的动态人体捕获
- Python OpenCV学习笔记之:图像直方图均衡化
- python OpenCV学习笔记(十三):图片梯度
- Opencv3.0-python的那些事儿:(八)、Opencv的直方图均衡化
- 详解python OpenCV学习笔记之直方图均衡化
- openCV—Python(8)—— 图像直方图及其直方图均衡化
- python opencv入门 直方图均衡化(22)
- OpenCV-Python教程(10、直方图均衡化)
- opencv_python学习笔记十三
- opencv_python学习笔记十三
- Python3与OpenCV3.3 图像处理(十三)--反射投影
- OpenCV-Python教程(10、直方图均衡化)
- python OpenCV学习笔记(二十二):直方图均衡化
- OpenCV-Python教程(10、直方图均衡化)
- windows下的python-opencv的配置
- 用Python和OpenCV创建一个图片搜索引擎的完整指南
- 相似性︱python+opencv实现pHash算法+hamming距离(simhash)(三)
- OpenCV3 Python语言实现 笔记6
- 自学笔记:用python和opencv进行摄像头的人脸检测