首页 最新 热门 推荐

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

OpenCV-图像饱和度

  • 25-03-03 21:21
  • 2017
  • 12540
blog.csdn.net

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

实现原理

       图像饱和度是指图像色彩的纯洁性,色彩的鲜艳程度,它是影响色彩最终效果的重要属性之一。饱和度也被称为图片色彩纯度,即色彩中彩色成分和消色成分的占比,这个比例决定了色彩的饱和度及鲜艳程度。当色彩中彩色成分多时,其色彩就呈现饱和(色觉强)、鲜明效果,给人的视觉印象会更强烈;反之,若消色成分多,色彩会显得暗淡,视觉效果也随之减弱。

       饱和度调整算法的实现流程如下:

       1.设置调整参数percent,取值为-100到100,类似PS中设置,归一化后为-1到1。

       2.针对图像所有像素点单个处理。计算RGB三通道的最大值最小值,可进一步得到delta和value:

\\delta=(Max-Min)/255 \\value=(Max+Min)/255

       3.若最大最小一致,即delta=0,则表明为灰点,不需继续操作,直接处理下个像素。

       4.通过value计算出HSL中的L值:

L=(Max-Min)/(2*255)

       5.S值为:

\left\{\begin{matrix} S=delta/value,L<0.5\\ S=delta/(2-value),L\geqslant 0.5 \end{matrix}\right.

       6.当percent大于等于0时,即提高色彩饱和度,那么alpha值为:

\left\{\begin{matrix} alpha=S,percent+S\geqslant 1\\ alpha=1-percent,else \end{matrix}\right.

       此时,调整后的图像RGB三通道值为:

RGB=RGB+(RGB-L*255)*alpha

       7.若percent小于0时,即降低色彩饱和度,则alpha=percent,此时调整后的图像RGB三通道值为:

RGB=L*255+(RGB-L*255)*(1+alpha)

       至此,图像实现了饱和度的调整,算法逻辑参考xingyanxiao。C++实现代码如下。

功能函数代码

  1. // 饱和度
  2. cv::Mat Saturation(cv::Mat src, int percent)
  3. {
  4. float Increment = percent* 1.0f / 100;
  5. cv::Mat temp = src.clone();
  6. int row = src.rows;
  7. int col = src.cols;
  8. for (int i = 0; i < row; ++i)
  9. {
  10. uchar *t = temp.ptr<uchar>(i);
  11. uchar *s = src.ptr<uchar>(i);
  12. for (int j = 0; j < col; ++j)
  13. {
  14. uchar b = s[3 * j];
  15. uchar g = s[3 * j + 1];
  16. uchar r = s[3 * j + 2];
  17. float max = max3(r, g, b);
  18. float min = min3(r, g, b);
  19. float delta, value;
  20. float L, S, alpha;
  21. delta = (max - min) / 255;
  22. if (delta == 0)
  23. continue;
  24. value = (max + min) / 255;
  25. L = value / 2;
  26. if (L < 0.5)
  27. S = delta / value;
  28. else
  29. S = delta / (2 - value);
  30. if (Increment >= 0)
  31. {
  32. if ((Increment + S) >= 1)
  33. alpha = S;
  34. else
  35. alpha = 1 - Increment;
  36. alpha = 1 / alpha - 1;
  37. t[3 * j + 2] =static_cast<uchar>( r + (r - L * 255) * alpha);
  38. t[3 * j + 1] = static_cast<uchar>(g + (g - L * 255) * alpha);
  39. t[3 * j] = static_cast<uchar>(b + (b - L * 255) * alpha);
  40. }
  41. else
  42. {
  43. alpha = Increment;
  44. t[3 * j + 2] = static_cast<uchar>(L * 255 + (r - L * 255) * (1 + alpha));
  45. t[3 * j + 1] = static_cast<uchar>(L * 255 + (g - L * 255) * (1 + alpha));
  46. t[3 * j] = static_cast<uchar>(L * 255 + (b - L * 255) * (1 + alpha));
  47. }
  48. }
  49. }
  50. return temp;
  51. }

C++测试代码

  1. #include
  2. using namespace cv;
  3. using namespace std;
  4. #define max2(a,b) (a>b?a:b)
  5. #define max3(a,b,c) (a>b?max2(a,c):max2(b,c))
  6. #define min2(a,b) (a
  7. #define min3(a,b,c) (a
  8. cv::Mat Saturation(cv::Mat src, int value);
  9. int main()
  10. {
  11. cv::Mat src = imread("House.jpg");
  12. cv::Mat result = Saturation(src, 100);
  13. imshow("original", src);
  14. imshow("result", result);
  15. waitKey(0);
  16. return 0;
  17. }
  18. // 饱和度
  19. cv::Mat Saturation(cv::Mat src, int percent)
  20. {
  21. float Increment = percent* 1.0f / 100;
  22. cv::Mat temp = src.clone();
  23. int row = src.rows;
  24. int col = src.cols;
  25. for (int i = 0; i < row; ++i)
  26. {
  27. uchar *t = temp.ptr(i);
  28. uchar *s = src.ptr(i);
  29. for (int j = 0; j < col; ++j)
  30. {
  31. uchar b = s[3 * j];
  32. uchar g = s[3 * j + 1];
  33. uchar r = s[3 * j + 2];
  34. float max = max3(r, g, b);
  35. float min = min3(r, g, b);
  36. float delta, value;
  37. float L, S, alpha;
  38. delta = (max - min) / 255;
  39. if (delta == 0)
  40. continue;
  41. value = (max + min) / 255;
  42. L = value / 2;
  43. if (L < 0.5)
  44. S = delta / value;
  45. else
  46. S = delta / (2 - value);
  47. if (Increment >= 0)
  48. {
  49. if ((Increment + S) >= 1)
  50. alpha = S;
  51. else
  52. alpha = 1 - Increment;
  53. alpha = 1 / alpha - 1;
  54. t[3 * j + 2] =static_cast( r + (r - L * 255) * alpha);
  55. t[3 * j + 1] = static_cast(g + (g - L * 255) * alpha);
  56. t[3 * j] = static_cast(b + (b - L * 255) * alpha);
  57. }
  58. else
  59. {
  60. alpha = Increment;
  61. t[3 * j + 2] = static_cast(L * 255 + (r - L * 255) * (1 + alpha));
  62. t[3 * j + 1] = static_cast(L * 255 + (g - L * 255) * (1 + alpha));
  63. t[3 * j] = static_cast(L * 255 + (b - L * 255) * (1 + alpha));
  64. }
  65. }
  66. }
  67. return temp;
  68. }

测试效果

图1 原图
图2 percent为50时的效果图
图3 percent为-50时的效果图

       通过调整percent可以实现图像饱和度的调整。

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

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

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

/ 登录

评论记录:

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

分类栏目

后端 (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