Matlab/C++源码实现RGB通道与HSV通道的转换(效果对比Halcon)

news2025/1/19 14:19:08

HSV通道的含义

HSV通道是指图像处理中的一种颜色模型,它由色调(Hue)、饱和度(Saturation)和明度(Value)三个通道组成。色调表示颜色的种类,饱和度表示颜色的纯度或鲜艳程度,明度表示颜色的亮度。HSV通道常用于图像处理中的颜色分析、颜色过滤、颜色调整等任务,它相对于其他颜色模型具有更直观和易于调节的特点,因此被广泛应用于计算机视觉和图像处理的领域。

Halcon算子例程

read_image (Image, 'D:/lena.jpg')
decompose3 (Image, ImageR, ImageG, ImageB)
trans_from_rgb (ImageR, ImageG, ImageB, ImageH, ImageS, ImageV, 'hsv')
trans_to_rgb (ImageH, ImageS, ImageV, ImageR1, ImageG1, ImageB1, 'hsv')
compose3 (ImageR1, ImageG1, ImageB1, MultiChannelImage)

这里先将三通道RGB三通道拆开成单独的通道,再将RGB与HSV通道互相转换,最后将三通道图像合并成RGB图像。
Halcon的图像效果是:
在这里插入图片描述

源代码实现

RGB转成HSV

在这里插入图片描述

这里需要注意的是,halcon这里将HSV三通道的取值范围作了说明,H通道的数值范围是0到2*pi,S通道的数值范围是0到1,V通道的数值范围是0到1。而常用的图像为BYTE字节型,数值范围是0到255,这里对公式做了修改,使Matlab得出的图像数据范围是0到255,可以直接显示,这里可以从matlab的workspace中看到计算过程。
以下便是使用Matlab实现trans_from_rgb的效果

trans_from_rgb (ImageR, ImageG, ImageB, ImageH, ImageS, ImageV, 'hsv')
Matlab源码
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%代码--RGB通道转HSV通道
%时间:2023.9
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
clc;
image=imread('D:\lena.jpg');
[height,width,channels]=size(image);
figure;
imshow(image);
title('rgb-image');
image_R=image(:,:,1);
image_R=double(image_R);
image_G=image(:,:,2);
image_G=double(image_G);
image_B=image(:,:,3);
image_B=double(image_B);

%%%转化成HSV通道
H_image = zeros(height,width);
S_image = zeros(height,width);
V_image = zeros(height,width);

 %%RGB转成HSV
for i=1:1:height
     for j=1:1:width
         %%%计算三通道的最大最小值计算
         image_matrix = [image_R(i,j), image_G(i,j), image_B(i,j)];
         maxValue = max(image_matrix);
         minValue = min(image_matrix);
         V_image(i,j) = maxValue;
         if(maxValue == minValue)
             S_image(i,j) = 0;
             H_image(i,j) = 0;
         else
              %%%计算饱和度
              S_image(i,j) = (maxValue - minValue)*255/minValue;
              %%%计算H通道
              if(maxValue == image_R(i,j))
                  H_image(i,j) = 42.5.*(image_G(i,j) - image_B(i,j))./(maxValue - minValue);
              elseif(maxValue == image_G(i,j))
                  H_image(i,j) = 42.5 * (2 + (image_B(i,j) - image_R(i,j)) / (maxValue - minValue));
              elseif(maxValue == image_B(i,j)) 
                  H_image(i,j) = 42.5 * (4 + (image_R(i,j) - image_G(i,j)) / (maxValue - minValue));
              end
         end
     end
end

 %%%RGB要取整
H_image = uint8(H_image);
S_image = uint8(S_image);
V_image = uint8(V_image);

figure;
imshow(H_image);
title('H_image');

figure;
imshow(S_image);
title('S_image');

figure;
imshow(V_image);
title('V_image');

最终实现的效果是:
在这里插入图片描述
最终验证的效果与halcon效果一致;

同时,以上代码采用C++实现的话如下所示,这里为了保证精度,输出结果采用的是double类型,但是范围也是0到255之间,要显示的话,需要转化为unsigned char类型:

C++源码
//将RGB图像转化成HSV图像
/*
输入:  rData : r通道图像
		gData : g通道图像
		bData : b通道图像
输出:  hDoubleData : h通道图像, h通道采用double类型,保留精度
		sDoubleData : s通道图像,s通道采用double类型,保留精度
		vDoubleData : v通道图像,v通道采用double类型,保留精度
*/
void trans_from_rgb(unsigned char *rData, unsigned char *gData, unsigned char *bData, double *hDoubleData, double *sDoubleData, double *vDoubleData, int height, int width)
{
	if ((height <= 0) || (width <= 0))
		return;
	//在函数外部分配好内存空间
	if (rData == NULL || gData == NULL || bData == NULL || hDoubleData == NULL || sDoubleData == NULL || vDoubleData == NULL)
		return;
	int i;
	unsigned char minValue,maxValue;
	for (i = 0; i < width * height; i++){
		//V通道数据,三通道的最大值
		maxValue = std::max(std::max(rData[i], gData[i]), bData[i]);
		minValue = std::min(std::min(rData[i], gData[i]), bData[i]);
		vDoubleData[i] = maxValue;
		if (maxValue == minValue){
			sDoubleData[i] = 0;
			hDoubleData[i] = 0;
		}
		else{
			//S通道
			sDoubleData[i] = (maxValue - minValue)*255.0 / maxValue;
			//H通道
			if (maxValue == rData[i])
				hDoubleData[i] = 42.5 * (gData[i] - bData[i]) / (maxValue - minValue);    
			else if (maxValue == gData[i])
				hDoubleData[i] = 42.5 * (2 + (bData[i] - rData[i]) / (maxValue - minValue));
			else if (maxValue == bData[i])
				hDoubleData[i] = 42.5 * (4 + (rData[i] - gData[i]) / (maxValue - minValue));
		}
	}
}
HSV转成RGB

halcon给出的公式说明为:
在这里插入图片描述

Matlab源码

同样的,采用Matlab实现:

%%%%%%%%%%%%%%%%%%%%%%%%%%%
%代码--HSV通道转RGB通道
%时间:2023.9
%%%%%%%%%%%%%%%%%%%%%%%%%%%%
clc;
image1=imread('D:\lena.jpg');
[height,width,channels]=size(image1);
image_H=double(image1);
image2=imread('E:\S_image.bmp');
image_S=double(image2);
image3=imread('E:\V_image.bmp');
image_V=double(image3);

%%%转化成RGB通道
R_image = zeros(height,width);
G_image = zeros(height,width);
B_image = zeros(height,width);

%%%HSV转成RGB
for i=1:1:height
     for j=1:1:width
        if(image_S(i,j) == 0)
            R_image(i,j) = image_V(i,j);
            G_image(i,j) = image_V(i,j);
            B_image(i,j) = image_V(i,j);
        else
            %%Hi = floor(image_H(i,j)*2*pi/255/deg2rad(60));    %%归一化到02*pi
            Hi = floor(image_H(i,j)*0.025);
            Hf = image_H(i,j)*0.025 - Hi;
            %%%%根据H的值,将C,X,m分别对应到RGB三个分量上
            if(Hi == 0)
                R_image(i,j) = image_V(i,j);
                G_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255*(1-Hf));
                B_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255);
            elseif(Hi == 1)
                R_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255*Hf);
                G_image(i,j) = image_V(i,j);
                B_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255);
            elseif(Hi == 2)
                R_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255);
                G_image(i,j) = image_V(i,j);
                B_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255*(1-Hf));
            elseif(Hi == 3)
                R_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255);
                G_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255*Hf);
                B_image(i,j) = image_V(i,j);
            elseif(Hi == 4)
                R_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255*(1-Hf));
                G_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255);
                B_image(i,j) = image_V(i,j);
            elseif(Hi == 5)
                R_image(i,j) = image_V(i,j);
                G_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255);
                B_image(i,j) = image_V(i,j) * (1 - image_S(i,j)/255*Hf);
            end
        end
     end
end

%%%RGB要取整
R_image = uint8(R_image);
G_image = uint8(G_image);
B_image = uint8(B_image);

figure;
imshow(R_image);
title('R_image');

figure;
imshow(G_image);
title('G_image');

figure;
imshow(B_image);
title('B_image');

最终实现的效果是:
在这里插入图片描述
可以看出,与Halcon效果一致;

C++源码

同样的,采用C++实现:

//将HSV图像转化成RGB图像
/*
返回:	NULL
作者:  祝春雨
时间:  2023.10.13
输入:  hDoubleData : h通道图像 ,h通道采用double类型,保留精度
		sDoubleData : s通道图像 ,s通道采用double类型,保留精度
		vDoubleData : v通道图像 ,v通道采用double类型,保留精度
输出:  rDoubleData : r通道图像 ,r通道采用double类型,保留精度
		gDoubleData : g通道图像 ,g通道采用double类型,保留精度
		bDoubleData : b通道图像 ,b通道采用double类型,保留精度
*/
void trans_to_rgb(double *hDoubleData, double *sDoubleData, double *vDoubleData, double *rDoubleData, double *gDoubleData, double *bDoubleData, int height, int width)
{
	if ((height <= 0) || (width <= 0))
		return;
	if (rDoubleData == NULL || gDoubleData == NULL || bDoubleData == NULL || hDoubleData == NULL || sDoubleData == NULL || vDoubleData == NULL)
		return;
	int i;
	double Hi, Hf;
	for (i = 0; i < width * height; i++){
		if (sDoubleData[i] > 0){
			Hi = floor(hDoubleData[i] * 0.025);
			Hf = hDoubleData[i] * 0.025 - Hi;
			if (Hi == 0){
				rDoubleData[i] = vDoubleData[i];
				gDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0*(1 - Hf));
				bDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0);
			}
			else if (Hi == 1){
				rDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0 * Hf);
				gDoubleData[i] = vDoubleData[i];
				bDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0);
			}
			else if (Hi == 2){
				rDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0);
				gDoubleData[i] = vDoubleData[i];
				bDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0*(1 - Hf));
			}
			else if (Hi == 3){
				rDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0);
				gDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0 * Hf);
				bDoubleData[i] = vDoubleData[i];
			}
			else if (Hi == 4){
				rDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0*(1 - Hf));
				gDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0);
				bDoubleData[i] = vDoubleData[i];
			}
			else if (Hi == 5){
				rDoubleData[i] = vDoubleData[i];
				gDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0);
				bDoubleData[i] = vDoubleData[i] * (1 - sDoubleData[i] / 255.0 * Hf);
			}
		}
		else{
			rDoubleData[i] = gDoubleData[i] = bDoubleData[i] = vDoubleData[i];
		}
	}
}

通道拆分与合并C++源代码实现

三通道拆分

Halcon中,拆分三通道的算子为:

decompose3 (Image, ImageR, ImageG, ImageB)

对应的拆分三通道图像的C++函数为:

//拆分三通道图像
/*
输入:  srcData : 三通道图像,内存排列方式是BGRBGRBGR......
输出:  rData : r通道图像
		gData : g通道图像
		bData : b通道图像
*/
void decompose3(unsigned char *srcData, unsigned char *rData, unsigned char *gData, unsigned char *bData, int height, int width)
{
	if ((height <= 0) || (width <= 0))
		return;
	if (srcData == NULL || rData == NULL || gData == NULL || bData == NULL)
		return;
	int i;
#pragma omp parallel for num_threads(3)
	for (i = 0; i < width * height; i++)
	{
		bData[i] = srcData[3 * i];
		gData[i] = srcData[3 * i + 1];
		rData[i] = srcData[3 * i + 2];
	}
}

三通道合并

Halcon中,拆分三通道的算子为:

compose3 (ImageR1, ImageG1, ImageB1, MultiChannelImage)

对应的合并三通道图像的C++函数为:

//合并三通道图像
/*
输入:  rData : r通道图像
		gData : g通道图像
		bData : b通道图像
输出:  bgrData:彩色图像,合并成BGRBGR.....排列
*/
void compose3(unsigned char *rData, unsigned char *gData, unsigned char *bData, unsigned char *bgrData, int height, int width)
{
	if ((height <= 0) || (width <= 0))
		return;
	if (bgrData == NULL || rData == NULL || gData == NULL || bData == NULL)
		return;
	int i;
#pragma omp parallel for num_threads(3)
	for (i = 0; i < width * height; i++)
	{
		bgrData[3 * i] = bData[i];
		bgrData[3 * i + 1] = gData[i];
		bgrData[3 * i + 2] = rData[i];
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1107258.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JAVA基础(JAVA SE)学习笔记(三)流程控制语句

前言 1. 学习视频&#xff1a; 尚硅谷Java零基础全套视频教程(宋红康2023版&#xff0c;java入门自学必备)_哔哩哔哩_bilibili 2023最新Java学习路线 - 哔哩哔哩 正文 JAVA基础&#xff08;JAVA SE&#xff09;学习笔记&#xff08;一&#xff09;JAVA学习路线、行业了解、…

Sql Server Report Service 使用简单说明

ReportServices做为报表服务器&#xff0c;结合sql直接访问数据库提供基本的报表格式设置显示&#xff0c;可以快速开发报表&#xff0c;主要包含两部分内容&#xff1a; 1.ReportServices服务器配置搭建&#xff0c;承载报表的运行平台 2.设计报表 ReportServices配置 1&am…

【C++初阶(二)】缺省参数函数重载

目录 前言 1. 缺省参数 1.1 什么是缺省参数 1.2 缺省参数的分类 1.2.1 全缺省参数 1.2.2 半缺省参数 2. 函数重载 2.1 什么是函数重载 2.2 缺省参数与函数重载 2.3 函数重载的使用 3. C支持函数重载的原因 总结 前言 在学习C语言时我们就会发现&#xff0c;C语言中存在的许多…

使用 Python 交互式方法预测股票价格变动概率

一、简介 当深入金融世界时,了解股价走势是一个显着的优势。在这里,我们提出了一种交互式方法,可以根据历史数据深入了解股价达到某些目标的可能性。 该工具利用 Python 强大的库,信息丰富且视觉上引人入胜。所提供的工具不会预测未来价格,而是评估价格波动的历史频率,让…

牛客网 -- WY28 跳石板

题目链接&#xff1a; 跳石板_牛客题霸_牛客网 (nowcoder.com) 解题步骤&#xff1a; 参考代码&#xff1a; void get_approximate(vector<int>& v,int n) {//求约数&#xff0c;从2到sqrt(n)即可&#xff0c;原因看图解//这里一定要等于sqrt(n)&#xff0c;例如16…

数据挖掘(6)聚类分析

一、什么是聚类分析 1.1概述 无指导的&#xff0c;数据集中类别未知类的特征&#xff1a; 类不是事先给定的&#xff0c;而是根据数据的相似性、距离划分的聚类的数目和结构都没有事先假定。挖掘有价值的客户: 找到客户的黄金客户ATM的安装位置 1.2区别 二、距离和相似系数 …

轻量级超分网络:Edge-oriented Convolution Block for Real-timeMM21_ECBSR 和 eSR

文章目录 ECBSR&#xff08;Edge-oriented Convolution Block for Real-timeMM21_ECBSR&#xff09;1. 作者目的是开发一个高效的适合移动端的超分网络。2. 作者决定使用plain net &#xff0c;但是效果不好&#xff0c;因此利用重参数化方法&#xff0c;丰富特征表示。3. re-p…

【数字人】6、ER-NeRF | 借助空间分解来实现基于 NeRF 的更高效的数字人生成(ICCV2023)

文章目录 一、背景二、方法2.1 问题设定2.2 Tri-Plane Hash Representation2.3 Region Attention Module2.4 训练细节 三、效果3.1 实验设定3.2 定量对比3.3 定性对比3.4 User study3.5 消融实验 四、代码4.1 视频数据预处理4.2 训练4.3 推理 论文&#xff1a;Efficient Region…

springboot+jaspersoft studio6制作报表

文章目录 前言一、开发工具下载安装二、开始制作1.新建1.文本2.图片3. 表格4.时间 三.遇到的问题1.中文不显示2.detail模块与column Footer之间有空白。怎么调节也消不掉 四.完整代码总结 前言 公司最近要做报销系统。需求就是将报销申请、报销审批。并将报销信息打印出来。 …

Nginx配置微服务避免actuator暴露

微服务一般在扫漏洞的情况下&#xff0c;需要屏蔽actuator健康检查 # 避免actuator暴露 if ($request_uri ~ "/actuator") { return 403; }

SD/SDIO(1):SD总线协议介绍

SD标准提供了很大的灵活性&#xff0c;除了作为存储卡外&#xff0c;还提供了SD卡槽的标准来扩展设备的功能。本篇文章就先来介绍一下SD总线的规范。对于SD/MMC协议的发展历史和概念介绍&#xff0c;可以参考我的这篇文章&#xff1a;SD、SDIO和MMC接口基础和规范介绍 文章目录…

【MySQL】数据库——库操作

文章目录 1. 创建数据库[IF NOT EXISTS] 的使用 2. 删除库3. 数据库的编码问题查看系统默认支持的字符集查看系统默认支持的校验集只查看 database的校验集指定编码创建数据库修改字符集修改校验集验证规则对数据库的影响utf8_general_ci ——不区分大小写utf8_bin ——区分大小…

【Qt之布局】QVBoxLayout、QHBoxLayout、QGridLayout、QFormLayout介绍及使用

在Qt中&#xff0c;布局管理器&#xff08;Layout&#xff09;用于管理窗口中的控件的位置和大小&#xff0c;以适应不同大小的窗口。 常用的布局管理器包括QVBoxLayout、QHBoxLayout、QGridLayout和QFormLayout。 先放张布局UI&#xff1a; 1. QVBoxLayout&#xff08;垂直布…

【爬虫教程】2023最详细的爬虫入门教程~

初识爬虫 学习爬虫之前&#xff0c;我们首先得了解什么是爬虫。 来自于百度百科的解释&#xff1a; 网络爬虫&#xff08;又称为网页蜘蛛&#xff0c;网络机器人&#xff0c;在FOAF社区中间&#xff0c;更经常的称为网页追逐者&#xff09;&#xff0c;是一种按照一定的规则&a…

视频批量加水印:保护版权,提升效率

在当今的自媒体时代&#xff0c;视频制作已经成为许多人的一项必备技能。然而&#xff0c;在视频制作过程中&#xff0c;如何为自己的视频添加独特的水印以保护知识产权&#xff0c;常常让许多制作者感到困扰。本文将为你揭示如何通过固乔剪辑助手软件&#xff0c;简单几步批量…

音乐播放器蜂鸣器ROM存储歌曲verilog,代码/视频

名称&#xff1a;音乐播放器蜂鸣器ROM存储歌曲 软件&#xff1a;Quartus 语言&#xff1a;Verilog 代码功能&#xff1a; 设计音乐播放器&#xff0c;要求至少包含2首歌曲&#xff0c;使用按键切换歌曲&#xff0c;使用开发板的蜂鸣器播放音乐&#xff0c;使用Quartus内的RO…

VUE前端判断是电脑端还是移动端

背景需求 ruoyi框架&#xff0c;前后端分离。现在要在用户访问的时候根据不同的设备跳转到不同的登录页面。 教程 router/index.js 修改src/router/index.js&#xff0c;在这里增加自己的要跳转的页面 permission.js 在白名单中添加自己的登录页面 增加以下识别的代码 le…

solidworks 2024新功能之-打造更加智能的工作 硕迪科技

SOLIDWORKS 2024 的新增功能 SOLIDWORKS 的每个版本都致力于改进您的工作流程&#xff0c;使您常用的工具尽可能快速高效地运作。此外&#xff0c;SOLIDWORKS 2024 可以通过量身定制的解决方案扩展您的工具集&#xff0c;并使您能够通过 Cloud Services 轻松将您的设计数据连接…

vue 写一个大富翁 和 老虎机组件

大富翁 老虎机https://github.com/YaminZheng/zillionaire.git Vue Ts 编写的大富翁&#xff0c;支持自定义路径&#xff0c;动画和图片可以自行添加 Dev git clone https://github.com/YaminZheng/zillionaire.git cd zillionaire yarn set version stable yarn install …

文心大模型4.0亮相2023百度世界大会,助力各赛道应用进一步发展

&#x1f3c6;作者简介&#xff0c;黑夜开发者&#xff0c;CSDN领军人物&#xff0c;全栈领域优质创作者✌&#xff0c;CSDN博客专家&#xff0c;阿里云社区专家博主&#xff0c;2023年6月CSDN上海赛道top4。 &#x1f3c6;数年电商行业从业经验&#xff0c;历任核心研发工程师…