使用颜色转换算法实现图像调色

2024-08-28 14:18

本文主要是介绍使用颜色转换算法实现图像调色,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在使用图像处理软件给图像调色时,我们通常会使用色彩平衡或曲线等功能,有时调来调去似乎总是不太满意。有没有一种方便、省事的方法,通过鼠标简单点选,就能将待处理的图片调成自己喜欢的某张图片色调呢?前几天在浏览SIGGRAPH 2013会议论文集时,发现几篇调色相关文章,但在算法实现步骤上作者似乎讲的不是很详细,实现起来有一定难度。在进一步查阅资料过程中,发现了这篇文章《Color Transfer between Images》,文章讲的很详细,算法实现起来比较容易。
算法流程大体为:
首先,将源图像和目标图像由rgb空间转为lab空间;
其次,分别计算源图像和目标图像均值及标准差;
然后,在lab空间,遍历每个像素点,分别对每个像素点计算各个通道值,计算方法为:将源图像减去自身均值后乘以标准差比值,再将相乘结果加上目标图像均值后赋值给输出图像;
最后,将输出图像由lab空间转为rgb空间。 至此,算法执行完毕。
可能有人会有疑惑,为什么要在lab空间完成整个计算过程,文章作者表述说:“rgb颜色空间各通道之间存在相关性,如果要改变一个像素颜色值,就必须改变三个颜色通道,这使得调色过程变得复杂,而lab颜色空间各通道之间相关性最小,可以在不同通道独立地进行颜色信息的统计和运算,同时lab颜色空间色域宽阔,并且与设备无关”。当然,在rgb空间也可以完成上述运算过程,经过实验,在rgb空间完成颜色转换后输出的色调看起来也不错,这个就仁者见仁智者见智了。
下面为算法的c++实现,程序中颜色空间转换过程直接调用了opencv相关函数,所以算法实现上容易了很多。
#include "stdafx.h"
#include <cv.h>
#include <cxcore.h>
#include <highgui.h>using namespace cv;#define max_uchar(a, b)    (((a) > (b)) ? (a) : (b))
#define min_uchar(a, b)    (((a) < (b)) ? (a) : (b))// 计算彩色图像均值和标准差
void CompMeanAndVariance(Mat &img, Vec3f &mean3f, Vec3f &variance3f)
{int row = img.rows;int col = img.cols;int total = row * col;float sum[3] = { 0.0f };// 均值uchar *pImg = img.data;for (int i = 0; i < row; i++){for (int j = 0; j < col; j++){sum[0] += pImg[3*j + 0];sum[1] += pImg[3*j + 1];sum[2] += pImg[3*j + 2];}pImg += img.step;}mean3f[0] = sum[0] / total;mean3f[1] = sum[1] / total;mean3f[2] = sum[2] / total;memset(sum, 0, sizeof(sum));// 标准差pImg = img.data;for (int i = 0; i < row; i++){for (int j = 0; j < col; j++){sum[0] += (pImg[3*j + 0] - mean3f[0]) * (pImg[3*j + 0] - mean3f[0]);sum[1] += (pImg[3*j + 1] - mean3f[1]) * (pImg[3*j + 1] - mean3f[1]);sum[2] += (pImg[3*j + 2] - mean3f[2]) * (pImg[3*j + 2] - mean3f[2]);}pImg += img.step;}variance3f[0] = sqrt(sum[0] / total);variance3f[1] = sqrt(sum[1] / total);variance3f[2] = sqrt(sum[2] / total);
}// 颜色转换
void ColorTransfer(Mat &src, Mat &tar, Mat &dst)
{Mat srcLab, tarLab;Vec3f srcMean3f, tarMean3f;// 源/目标图像均值Vec3f srcVariance3f, tarVariance3f;// 源/目标图像标准差Vec3f ratioVariance3f;// 标准差比例// BGR空间转Lab空间cvtColor(src, srcLab, CV_BGR2Lab);cvtColor(tar, tarLab, CV_BGR2Lab);// 计算当前图像与目标图像均值及标准差CompMeanAndVariance(srcLab, srcMean3f, srcVariance3f);CompMeanAndVariance(tarLab, tarMean3f, tarVariance3f);// 标准差比ratioVariance3f[0] = tarVariance3f[0] / srcVariance3f[0];ratioVariance3f[1] = tarVariance3f[1] / srcVariance3f[1];ratioVariance3f[2] = tarVariance3f[2] / srcVariance3f[2];// 计算颜色转换值int row = srcLab.rows;int col = srcLab.cols;uchar *pImg = srcLab.data;for (int i = 0; i < row; i++){for (int j = 0; j < col; j++){pImg[3*j + 0] = (uchar)min_uchar(255, max_uchar(0, ratioVariance3f[0] * (pImg[3*j + 0] - srcMean3f[0]) + tarMean3f[0]));pImg[3*j + 1] = (uchar)min_uchar(255, max_uchar(0, ratioVariance3f[1] * (pImg[3*j + 1] - srcMean3f[1]) + tarMean3f[1]));pImg[3*j + 2] = (uchar)min_uchar(255, max_uchar(0, ratioVariance3f[2] * (pImg[3*j + 2] - srcMean3f[2]) + tarMean3f[2]));}pImg += srcLab.step;}// Lab空间转BGR空间cvtColor(srcLab, dst, CV_Lab2BGR);
}int main()
{Mat src = imread("image\\src.jpg");Mat tar = imread("image\\tar.jpg");imshow("src", src);imshow("tar", tar);// 调色Mat dst;ColorTransfer(src, tar, dst);imshow("dst", dst);waitKey();return 0;
}

工程下载链接: http://download.csdn.net/detail/u013085897/6801879
程序基于vs2005 + opencv210实现,下载工程后,如果与自己使用的opencv版本不一致,则需要对工程进行简单配置才能正确运行。

这篇关于使用颜色转换算法实现图像调色的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115004

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置