首页 最新 热门 推荐

  • 首页
  • 最新
  • 热门
  • 推荐

OpenCV-图像高光调整

  • 25-03-03 21:21
  • 2758
  • 13791
blog.csdn.net

作者:翟天保Steven
版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处

实现原理

       PS中的高光命令是一种校正由于太接近相机闪光灯而有些发白的焦点的方法。在用其他方式采光的图像中,这种调整也可用于使高光区域变暗。要实现图像的高光调整,首先要识别出高光区;再通过对高光区的色彩进行一定变换,使其达到提光或者暗化效果;最后也是最重要的,就是对高光区和非高光区的边缘作平滑处理。

       下方介绍具体流程。

具体流程

1)读取识别图像的原图,并转灰度图,再归一化。

  1. // 生成灰度图
  2. cv::Mat gray = cv::Mat::zeros(input.size(), CV_32FC1);
  3. cv::Mat f = input.clone();
  4. f.convertTo(f, CV_32FC3);
  5. vector<cv::Mat> pics;
  6. split(f, pics);
  7. gray = 0.299f*pics[2] + 0.587*pics[1] + 0.114*pics[0];
  8. gray = gray / 255.f;
图1 灰度图

2)确定高光区。因为我们要识别高光,所以thresh通过gray*gray,得到的图像中原本亮的地方则为亮,取平均值当阈值,进行二值化得到掩膜mask。

  1. // 确定高光区
  2. cv::Mat thresh = cv::Mat::zeros(gray.size(), gray.type());
  3. thresh = gray.mul(gray);
  4. // 取平均值作为阈值
  5. Scalar t = mean(thresh);
  6. cv::Mat mask = cv::Mat::zeros(gray.size(), CV_8UC1);
  7. mask.setTo(255, thresh >= t[0]);
图2 掩膜图

3)对掩膜区边缘进行平滑过渡。假设light为50,那么midrate的掩膜区值为1.5,黑色区为1,过渡区为1~1.5;bright的掩膜区为0.125,黑色区为0,过渡区为0~0.125。

  1. // 参数设置
  2. int max = 4;
  3. float bright = light / 100.0f / max;
  4. float mid = 1.0f + max * bright;
  5. // 边缘平滑过渡
  6. cv::Mat midrate = cv::Mat::zeros(input.size(), CV_32FC1);
  7. cv::Mat brightrate = cv::Mat::zeros(input.size(), CV_32FC1);
  8. for (int i = 0; i < input.rows; ++i)
  9. {
  10. uchar *m = mask.ptr<uchar>(i);
  11. float *th = thresh.ptr<float>(i);
  12. float *mi = midrate.ptr<float>(i);
  13. float *br = brightrate.ptr<float>(i);
  14. for (int j = 0; j < input.cols; ++j)
  15. {
  16. if (m[j] == 255)
  17. {
  18. mi[j] = mid;
  19. br[j] = bright;
  20. }
  21. else {
  22. mi[j] = (mid - 1.0f) / t[0] * th[j] + 1.0f;
  23. br[j] = (1.0f / t[0] * th[j])*bright;
  24. }
  25. }
  26. }

4)根据midrate和brightrate,进行高光区提亮。对非高光区而言,midrate都为1,brightrate都为0,即没有变化;对高光区而言,midrate都为1.5,brightrate都为0.125,所以色彩数值均有所增加,带来了提亮效果;对边缘地区,midrate和brightrate起到了很好的过渡作用。

注意:temp要进行数值限制,假设temp大于1,则进行uchar处理后数值会因为类型原因产生突变,那么图像也就变成了五颜六色。

  1. // 高光提亮,获取结果图
  2. cv::Mat result = cv::Mat::zeros(input.size(), input.type());
  3. for (int i = 0; i < input.rows; ++i)
  4. {
  5. float *mi = midrate.ptr<float>(i);
  6. float *br = brightrate.ptr<float>(i);
  7. uchar *in = input.ptr<uchar>(i);
  8. uchar *r = result.ptr<uchar>(i);
  9. for (int j = 0; j < input.cols; ++j)
  10. {
  11. for (int k = 0; k < 3; ++k)
  12. {
  13. float temp = pow(float(in[3 * j + k]) / 255.f, 1.0f / mi[j])*(1.0 / (1 - br[j]));
  14. if (temp > 1.0f)
  15. temp = 1.0f;
  16. if (temp < 0.0f)
  17. temp = 0.0f;
  18. uchar utemp = uchar(255*temp);
  19. r[3 * j + k] = utemp;
  20. }
  21. }
  22. }

C++测试代码

  1. #include <iostream>
  2. #include <opencv2/opencv.hpp>
  3. using namespace std;
  4. using namespace cv;
  5. cv::Mat HighLight(cv::Mat input, int light);
  6. int main()
  7. {
  8. cv::Mat src = imread("test3.jpg");
  9. int light1 = 50;
  10. int light2 = -50;
  11. cv::Mat result1 = HighLight(src, light1);
  12. cv::Mat result2 = HighLight(src, light2);
  13. imshow("original", src);
  14. imshow("result1", result1);
  15. imshow("result2", result2);
  16. waitKey(0);
  17. return 0;
  18. }
  19. // 图像高光选取
  20. cv::Mat HighLight(cv::Mat input, int light)
  21. {
  22. // 生成灰度图
  23. cv::Mat gray = cv::Mat::zeros(input.size(), CV_32FC1);
  24. cv::Mat f = input.clone();
  25. f.convertTo(f, CV_32FC3);
  26. vector<cv::Mat> pics;
  27. split(f, pics);
  28. gray = 0.299f*pics[2] + 0.587*pics[1] + 0.114*pics[0];
  29. gray = gray / 255.f;
  30. // 确定高光区
  31. cv::Mat thresh = cv::Mat::zeros(gray.size(), gray.type());
  32. thresh = gray.mul(gray);
  33. // 取平均值作为阈值
  34. Scalar t = mean(thresh);
  35. cv::Mat mask = cv::Mat::zeros(gray.size(), CV_8UC1);
  36. mask.setTo(255, thresh >= t[0]);
  37. // 参数设置
  38. int max = 4;
  39. float bright = light / 100.0f / max;
  40. float mid = 1.0f + max * bright;
  41. // 边缘平滑过渡
  42. cv::Mat midrate = cv::Mat::zeros(input.size(), CV_32FC1);
  43. cv::Mat brightrate = cv::Mat::zeros(input.size(), CV_32FC1);
  44. for (int i = 0; i < input.rows; ++i)
  45. {
  46. uchar *m = mask.ptr<uchar>(i);
  47. float *th = thresh.ptr<float>(i);
  48. float *mi = midrate.ptr<float>(i);
  49. float *br = brightrate.ptr<float>(i);
  50. for (int j = 0; j < input.cols; ++j)
  51. {
  52. if (m[j] == 255)
  53. {
  54. mi[j] = mid;
  55. br[j] = bright;
  56. }
  57. else {
  58. mi[j] = (mid - 1.0f) / t[0] * th[j] + 1.0f;
  59. br[j] = (1.0f / t[0] * th[j])*bright;
  60. }
  61. }
  62. }
  63. // 高光提亮,获取结果图
  64. cv::Mat result = cv::Mat::zeros(input.size(), input.type());
  65. for (int i = 0; i < input.rows; ++i)
  66. {
  67. float *mi = midrate.ptr<float>(i);
  68. float *br = brightrate.ptr<float>(i);
  69. uchar *in = input.ptr<uchar>(i);
  70. uchar *r = result.ptr<uchar>(i);
  71. for (int j = 0; j < input.cols; ++j)
  72. {
  73. for (int k = 0; k < 3; ++k)
  74. {
  75. float temp = pow(float(in[3 * j + k]) / 255.f, 1.0f / mi[j])*(1.0 / (1 - br[j]));
  76. if (temp > 1.0f)
  77. temp = 1.0f;
  78. if (temp < 0.0f)
  79. temp = 0.0f;
  80. uchar utemp = uchar(255*temp);
  81. r[3 * j + k] = utemp;
  82. }
  83. }
  84. }
  85. return result;
  86. }

测试效果

图1 原图
图2 light为50的效果图
图3 light为-50的效果图

       从测试效果中可以看出,高光区随light变化而产生亮度变化,当light为正值时,高光区有明显提亮效果;反之,则变更暗。

       如果函数有什么可以改进完善的地方,非常欢迎大家指出,一同进步何乐而不为呢~

       如果文章帮助到你了,可以点个赞让我知道,我会很快乐~加油!

注:本文转载自blog.csdn.net的翟天保Steven的文章"https://zhaitianbao.blog.csdn.net/article/details/120360224"。版权归原作者所有,此博客不拥有其著作权,亦不承担相应法律责任。如有侵权,请联系我们删除。
复制链接
复制链接
相关推荐
发表评论
登录后才能发表评论和回复 注册

/ 登录

评论记录:

未查询到任何数据!
回复评论:

分类栏目

后端 (14832) 前端 (14280) 移动开发 (3760) 编程语言 (3851) Java (3904) Python (3298) 人工智能 (10119) AIGC (2810) 大数据 (3499) 数据库 (3945) 数据结构与算法 (3757) 音视频 (2669) 云原生 (3145) 云平台 (2965) 前沿技术 (2993) 开源 (2160) 小程序 (2860) 运维 (2533) 服务器 (2698) 操作系统 (2325) 硬件开发 (2492) 嵌入式 (2955) 微软技术 (2769) 软件工程 (2056) 测试 (2865) 网络空间安全 (2948) 网络与通信 (2797) 用户体验设计 (2592) 学习和成长 (2593) 搜索 (2744) 开发工具 (7108) 游戏 (2829) HarmonyOS (2935) 区块链 (2782) 数学 (3112) 3C硬件 (2759) 资讯 (2909) Android (4709) iOS (1850) 代码人生 (3043) 阅读 (2841)

热门文章

101
推荐
关于我们 隐私政策 免责声明 联系我们
Copyright © 2020-2025 蚁人论坛 (iYenn.com) All Rights Reserved.
Scroll to Top