目标
本章节我们将:
- 学习如何从曝光序列生成和显示 HDR 图像。
- 使用曝光融合来合并曝光序列。
理论
高动态范围成像 (HDRI 或 HDR) 是一种在成像和摄影中使用的技术,用于再现比标准数字成像或摄影技术所能实现的更大的亮度动态范围。虽然人眼可以适应各种光照条件,但大多数成像设备每个通道使用 8 位,因此我们仅限于 256 个级别。当我们拍摄现实场景的照片时,明亮区域可能会过曝,而黑暗区域可能会欠曝,因此我们无法使用单次曝光捕捉所有细节。HDR 成像使用每个通道超过 8 位(通常为 32 位浮点值)的图像,从而允许更大的动态范围。
获得 HDR 图像的方法有很多,但最常见的方法是使用以不同曝光值拍摄的场景照片。为了组合这些曝光,了解相机的响应函数非常有用,并且存在估计它的算法。合并 HDR 图像后,必须将其转换回 8 位才能在普通显示器上查看。此过程称为色调映射。当场景中的物体或相机在拍摄之间移动时,会产生额外的复杂性,因为应该对不同曝光的图像进行配准和对齐。
在本教程中,我们将展示两种算法(Debevec,Robertson)用于从曝光序列生成和显示 HDR 图像,并演示一种称为曝光融合(Mertens)的替代方法,该方法生成低动态范围图像,并且不需要曝光时间数据。此外,我们还估计了相机响应函数 (CRF),这对于许多计算机视觉算法都具有重要价值。HDR 管道的每个步骤都可以使用不同的算法和参数来实现,因此请查看参考手册以查看所有参数。
曝光序列 HDR
在本教程中,我们将查看以下场景,其中我们有 4 张曝光图像,曝光时间分别为:15 秒、2.5 秒、1/4 秒和 1/30 秒。(您可以从维基百科下载这些图像)
图像
1. 将曝光图像加载到列表中
第一步是简单地将所有图像加载到列表中。此外,对于常规 HDR 算法,我们将需要曝光时间。请注意数据类型,因为图像应该是 1 通道或 3 通道 8 位 (np.uint8),而曝光时间需要是 float32 并以秒为单位。
import cv2 as cv
import numpy as np
img_fn = ["img0.jpg", "img1.jpg", "img2.jpg", "img3.jpg"]
exposure_times = np.array([15.0, 2.5, 0.25, 0.0333], dtype=np.float32)
CV_EXPORTS_W Mat imread(const String &filename, int flags=IMREAD_COLOR_BGR)
从文件中加载图像。
2. 将曝光合并到 HDR 图像中
在此阶段,我们将曝光序列合并到一个 HDR 图像中,展示了 OpenCV 中的两种可能性。第一种方法是 Debevec,第二种方法是 Robertson。请注意,HDR 图像的类型为 float32,而不是 uint8,因为它包含所有曝光图像的完整动态范围。
hdr_debevec = merge_debevec.process(img_list, times=exposure_times.copy())
hdr_robertson = merge_robertson.process(img_list, times=exposure_times.copy())
Ptr< MergeDebevec > createMergeDebevec()
创建 MergeDebevec 对象。
Ptr< MergeRobertson > createMergeRobertson()
创建 MergeRobertson 对象。
3. 色调映射 HDR 图像
我们将 32 位浮点 HDR 数据映射到 [0..1] 范围。实际上,在某些情况下,值可能大于 1 或小于 0,因此请注意,稍后我们将必须剪辑数据以避免溢出。
res_debevec = tonemap1.process(hdr_debevec.copy())
Ptr< Tonemap > createTonemap(float gamma=1.0f)
创建一个具有伽马校正的简单线性映射器。
4. 使用 Mertens 融合合并曝光
在这里,我们展示了一种合并曝光图像的替代算法,我们不需要曝光时间。我们也不需要使用任何色调映射算法,因为 Mertens 算法已经为我们提供了 [0..1] 范围内的结果。
res_mertens = merge_mertens.process(img_list)
Ptr< MergeMertens > createMergeMertens(float contrast_weight=1.0f, float saturation_weight=1.0f, float exposure_weight=0.0f)
创建 MergeMertens 对象。
5. 转换为 8 位并保存
为了保存或显示结果,我们需要将数据转换为 [0..255] 范围内的 8 位整数。
res_debevec_8bit = np.clip(res_debevec*255, 0, 255).astype('uint8')
res_robertson_8bit = np.clip(res_robertson*255, 0, 255).astype('uint8')
res_mertens_8bit = np.clip(res_mertens*255, 0, 255).astype('uint8')
cv.imwrite(
"ldr_robertson.jpg", res_robertson_8bit)
cv.imwrite(
"fusion_mertens.jpg", res_mertens_8bit)
CV_EXPORTS_W bool imwrite(const String &filename, InputArray img, const std::vector< int > ¶ms=std::vector< int >())
将图像保存到指定文件。
结果
您可以看到不同的结果,但请考虑每个算法都有额外的参数,您应该调整这些参数以获得所需的结果。最佳实践是尝试不同的方法,并查看哪种方法最适合您的场景。
Debevec
图像
Robertson
图像
Mertenes 融合
图像
估计相机响应函数
相机响应函数 (CRF) 给出了场景辐射与测量强度值之间的关系。CRF 在一些计算机视觉算法中非常重要,包括 HDR 算法。在这里,我们估计相机响应函数的反函数,并将其用于 HDR 合并。
crf_debevec = cal_debevec.process(img_list, times=exposure_times)
hdr_debevec = merge_debevec.process(img_list, times=exposure_times.copy(), response=crf_debevec.copy())
crf_robertson = cal_robertson.process(img_list, times=exposure_times)
hdr_robertson = merge_robertson.process(img_list, times=exposure_times.copy(), response=crf_robertson.copy())
Ptr< CalibrateRobertson > createCalibrateRobertson(int max_iter=30, float threshold=0.01f)
创建 CalibrateRobertson 对象。
Ptr< CalibrateDebevec > createCalibrateDebevec(int samples=70, float lambda=10.0f, bool random=false)
创建 CalibrateDebevec 对象。
相机响应函数由每个颜色通道的 256 长度向量表示。对于此序列,我们得到以下估计值
图像
附加资源
- Paul E Debevec 和 Jitendra Malik。从照片中恢复高动态范围辐射图。在 ACM SIGGRAPH 2008 课程中,第 31 页。ACM,2008。[68]
- Mark A Robertson、Sean Borman 和 Robert L Stevenson。通过多次曝光提高动态范围。在图像处理中,1999 年。ICIP 99。会议论文集。1999 年国际会议,第 3 卷,第 159-163 页。IEEE,1999。[227]
- Tom Mertens、Jan Kautz 和 Frank Van Reeth。曝光融合。在计算机图形和应用中,2007 年。PG'07。第 15 届太平洋会议,第 382-390 页。IEEE,2007。[188]
- 图片来自 Wikipedia-HDR
练习
- 尝试所有色调映射算法:cv::TonemapDrago、cv::TonemapMantiuk 和 cv::TonemapReinhard
- 尝试更改 HDR 校准和色调映射方法中的参数。