CIE颜色空间LCh、Lab、XYZ介绍与转换关系(包含源码)

news2024/11/23 10:46:25

项目场景:

提示:在颜色科学中,LCh和Lab是比较常用的

LCh是由MATLAB计算出的数据,但是我所需要在Qt的q3dsurface绘制出这个切面,看了Qt官方Examples,墨西哥草帽算法的3D模型就是由XYZ组成的。所以我需要LCh->Lab->XYZ,这三步的转换。


LCh、Lab、XYZ概念

  • LCh
    CIELCh颜色空间用于描述颜色的亮度、色度和色相。
    CIELCh颜色空间相比CIELAB颜色空间更加直观,它以极坐标形式表示颜色属性,更适合描述颜色的外观特征和感知属性。
    Lightness(L)指的是颜色的明暗程度或亮度级别。它表示颜色相对于中性灰色的明暗程度,取值范围通常为0到100。较低的亮度值接近黑色,较高的亮度值接近白色。
    Chroma(C)表示颜色的饱和度或色彩的强度。它衡量的是颜色相对于中性灰色的纯度或饱和度。较低的色度值表示颜色较暗或接近灰色,而较高的色度值表示颜色鲜艳、饱和度较高。
    Hue表示色相(h),取值范围为0到360度,表示颜色在色轮上的位置。

  • Lab
    CIELAB颜色空间是通过对人眼对不同光谱刺激的感知进行数学建模而得到的。
    L表示亮度(Lightness),取值范围为0到100,表示从黑到白的亮度级别。
    a表示颜色在红绿轴上的位置,取值范围为-128到+127,其中负值表示绿色,正值表示红色。
    b表示颜色在黄蓝轴上的位置,取值范围为-128到+127,其中负值表示蓝色,正值表示黄色。

  • XYZ
    X表示颜色在红-绿轴上的位置。
    Y表示颜色在亮度轴上的位置。
    Z表示颜色在黄-蓝轴上的位置。


LCh转Lab

  1. 获取LCh颜色空间中的L、Chroma和Hue分量的值。
  2. 计算Lab颜色空间中的a和b分量,可以使用以下公式:a = Chroma * cos(Hue),b = Chroma * sin(Hue)。
  3. 最终的Lab颜色空间中的分量为L、a和b。
#include <cmath>

void LChToLab(double L, double Chroma, double Hue, double& Lab_L, double& Lab_a, double& Lab_b)
{
    Lab_a = Chroma * std::cos(Hue);
    Lab_b = Chroma * std::sin(Hue);
    Lab_L = L;
}

L表示LCh颜色空间中的亮度分量,Chroma表示色度分量,Hue表示色相分量。通过调用该函数并传入LCh颜色的分量值,即可获得对应的Lab颜色空间中的L、a和b分量的值


Lab转XYZ

  1. 获取Lab颜色空间中的L、a和b分量的值。
  2. 将Lab中的L、a和b分量进行逆变换,得到对应的线性RGB颜色空间中的R、G和B分量。这可以通过使用Lab到XYZ转换矩阵来实现。
  3. 将线性RGB颜色空间中的R、G和B分量进行非线性校正,以获得标准化的RGB值。
  4. 将标准化的RGB值转换为XYZ颜色空间中的X、Y和Z分量,也可以通过使用RGB到XYZ转换矩阵来实现。
void LabToXYZ(double Lab_L, double Lab_a, double Lab_b, double& XYZ_X, double& XYZ_Y, double& XYZ_Z)
{
    // Lab到XYZ转换矩阵
    double Xr = 0.95047;
    double Yr = 1.00000;
    double Zr = 1.08883;

    double fy = (Lab_L + 16.0) / 116.0;
    double fx = fy + (Lab_a / 500.0);
    double fz = fy - (Lab_b / 200.0);

    double xr = (std::pow(fx, 3.0) > 0.008856) ? std::pow(fx, 3.0) : ((fx - 16.0 / 116.0) / 7.787);
    double yr = (Lab_L > (903.3 * 0.008856)) ? std::pow((Lab_L + 16.0) / 116.0, 3.0) : (Lab_L / 903.3);
    double zr = (std::pow(fz, 3.0) > 0.008856) ? std::pow(fz, 3.0) : ((fz - 16.0 / 116.0) / 7.787);

    XYZ_X = xr * Xr;
    XYZ_Y = yr * Yr;
    XYZ_Z = zr * Zr;
}

在函数中,Lab_L、Lab_a和Lab_b分别表示Lab颜色空间中的L、a和b分量的值。通过调用该函数并传入Lab颜色的分量值,即可获得对应的XYZ颜色空间中的X、Y和Z分量的值。请注意,示例中的转换矩阵参数(Xr、Yr和Zr)是用于D65标准光源的参数,你可以根据需要调整它们以适应不同的光源。

Demo

#include <iostream>
#include <vector>
#include <cmath>

struct Lab {
    int L;
    int a;
    int b;
};

struct XYZ {
    int X;
    int Y;
    int Z;
};

XYZ convertLabtoXYZ(const Lab& lab)
{
    double L = lab.L / 100.0;
    double a = (lab.a - 128) / 127.0;
    double b = (lab.b - 128) / 127.0;

    double X = 0.95047 * pow(((L + 0.16) / 1.16), 3.0);
    double Y = 1.00000 * pow(((L + 0.16) / 1.16), 3.0);
    double Z = 1.08883 * pow(((L + 0.16) / 1.16), 3.0);

    X = X + (0.43607 * a) - (0.39894 * b);
    Y = Y - (0.00003 * a) + (0.03951 * b);
    Z = Z + (0.00816 * a) + (0.01388 * b);

    XYZ xyz;
    xyz.X = static_cast<int>(X * 100);
    xyz.Y = static_cast<int>(Y * 100);
    xyz.Z = static_cast<int>(Z * 100);

    return xyz;
}

int main()
{
    // 示例使用
    int N = 3;
    std::vector<Lab> labValues(N);
    std::vector<XYZ> xyzValues(N);

    // 假设输入的LAB值
    labValues[0] = {50, 0, 0};
    labValues[1] = {75, 30, -40};
    labValues[2] = {90, -10, 20};

    // 将每个LAB值转换为XYZ值
    for (int i = 0; i < N; ++i) {
        xyzValues[i] = convertLabtoXYZ(labValues[i]);
    }

    // 打印转换后的XYZ值
    for (int i = 0; i < N; ++i) {
        XYZ xyz = xyzValues[i];
        std::cout << "Lab: L=" << labValues[i].L << ", a=" << labValues[i].a << ", b=" << labValues[i].b
                  << " -> XYZ: X=" << xyz.X << ", Y=" << xyz.Y << ", Z=" << xyz.Z << std::endl;
    }

    return 0;
}

这边举了3组数据进行计算。
在这里插入图片描述
完结撒花,球球一件三联噢,这真的对我很重要
在这里插入图片描述

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

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

相关文章

【c语言】组件化打包—静态库

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞 关注支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; &#x1f525;c语言系列专栏&#xff1a;c语言之路重点知识整合 &#x…

MacOS使用Parallels Desktop安装win11虚拟机

文章目录 前话一、安装Parallels Desktop二、安装Windows11虚拟机1.win11镜像下载2.虚拟机启动 三、创建过程问题解决1.安装win11系统的时候不显示网路2.系统安装完成后无法连接网络 三、参考文献 前话 为了让大家能快速获取到需要的内容&#xff0c;这里提前说明一下本文适用…

杂记——23.java中的值传递和应用传递

这篇文章我们来讲一下java中的值传递和引用传递 结论&#xff1a;java中只存在值传递&#xff0c;不存在引用传递&#xff08;C中有引用传递&#xff09; 分析&#xff1a; 值传递(pass by value)&#xff1a;在调用函数时&#xff0c;将实际参数复制一份传递到函数中&#…

用DevExpress WinForms富文本编辑器,集成高级文本编辑功能(二)

DevExpress WinForm富文本编辑器&#xff08;RTF编辑器&#xff09;控件允许用户将高级文本编辑功能集成到下一个WinForms项目中&#xff0c;它包括全面的文本格式选项、支持邮件合并&#xff0c;并附带了丰富的终端用户选项集&#xff0c;因此可以轻松交付受Microsoft word启发…

助力工业物联网,工业大数据之数仓维度层DWS层构建【十二】

文章目录 数仓维度层DWS层构建01&#xff1a;项目回顾02&#xff1a;项目目标03&#xff1a;维度建模回顾&#xff1a;建模流程04&#xff1a;维度建模回顾&#xff1a;维度设计05&#xff1a;维度建模回顾&#xff1a;维度模型 数仓维度层DWS层构建 01&#xff1a;项目回顾 O…

《现代中学生》期刊简介及投稿要求

《现代中学生》期刊简介及投稿要求 《现代中学生》 主管单位&#xff1a;吉林省教育厅 主办单位&#xff1a; 吉林教育杂志社 国际刊号ISSN&#xff1a;1009-5748&#xff1b;国内刊号CN&#xff1a;22-1046/G4&#xff1b;邮发代号&#xff1a;12-52 出版周期&#xff1a…

刷题day65:整数拆分

题意描述&#xff1a; 给定一个正整数 n &#xff0c;将其拆分为 k 个 正整数 的和&#xff08; k > 2 &#xff09;&#xff0c;并使这些整数的乘积最大化。 返回 你可以获得的最大乘积 。. 思路&#xff1a; 首先dp[i]代表将i拆分&#xff0c;并得到的最大乘积&#xf…

springboot+jsp高校招生宣传系统java校园网站

学校招生管理系统的用户是系统最根本使用者&#xff0c;按需要分析系统包括两类用户&#xff1a;学生、管理员。这两类用户对系统的功能简要如下。 管理员通过后台的登录页面&#xff0c;选择管理员权限后进行登录&#xff0c;管理员的权限包括招生计划管理、招生录取信息管理和…

知识变现海哥:我们不生产知识,我们只是知识的搬运工!

现在大家都在谈知识付费&#xff0c;而且有一些东西就是书本上有的&#xff0c;但被录制成视频&#xff01;加上专业的讲师去讲&#xff0c;配上PPT文档&#xff0c;然后发布出去&#xff0c;标上一定的价格后&#xff0c;就会有人去付费购买学习。 想想是不是很暴利&#xff1…

如何使用ArcGIS生成随机数

&#xff08;本文首发于“水经注GIS”公号&#xff0c;关注公号免费领取地图数据&#xff09; 在制图的过程中&#xff0c;制作某些效果需要根据字段的随机数来生成对应的颜色&#xff0c;所以需要操作字段生成随机数&#xff0c;这里为大家介绍一下ArcGIS生成随机数的方法&…

对话金现代|轻骑兵低代码开发平台,破局企业“数字化焦虑

当今世界正经历百年未有之大变局&#xff0c;大数据、云计算、区块链、人工智能等新技术飞速发展&#xff0c;越来越多的企业经营者&#xff0c;开始投入到数字化转型的浪潮中&#xff0c;期望通过数字化帮助企业实现业务增长&#xff0c;开启无限可能。但随着时间的推移&#…

基于平台视角下的数据交易模式及特点分析

基于平台视角下的数据交易模式及特点分析 陈宏民1, 熊红林1, 胥莉1&#xff0c;杨云鹏1, 卓训方2 1 上海交通大学安泰经济与管理学院&#xff0c;上海 200030 2 上海数据交易所有限公司&#xff0c;上海 201203 摘要&#xff1a;基于国内外数据交易市场的发展现状与平台思维视角…

用GDB调试cmake编译后的C++程序

文章目录 编译时添加参数用GDB运行程序参考 编译时添加参数 cmake -DCMAKE_BUILD_TYPEDebug <path and other arguments>用GDB运行程序 开始调试 start设置断点&#xff0c;这样会在运行到源码第6行时停止&#xff0c;可以查看变量的值、堆栈情况等&#xff1b;这个行…

71.建立一个轮播图组件第二部分

这节课我们通过绝对定位的方式来实现上一届未实现的内容&#xff01; ● 首先我们先设置一下button和图标 /* CONTROLS */.btn {background-color: #fff;border: none;height: 40px;width: 40px;}.btn-icon {height: 24px;width: 24px;}● 接着我们来设置绝对定位 .carousel {p…

如何在 FPGA 中做数学运算

FPGA 非常适合进行数学运算&#xff0c;但是需要一点技巧&#xff0c;所以我们今天就看看如何在 FPGA 中进行简单和复杂的数学运算。 介绍 由于FPGA可以对算法进行并行化&#xff0c;所以FPGA 非常适合在可编程逻辑中实现数学运算。我们可以在 FPGA 中使用数学来实现信号处理、…

Java进阶-面向对象进阶(static和继承)

1 复习回顾 1.1 如何定义类 类的定义格式如下: 修饰符 class 类名 {// 1.成员变量&#xff08;属性&#xff09;// 2.成员方法 (行为) // 3.构造方法 &#xff08;初始化类的对象数据的&#xff09; }例如: public class Student {// 1.成员变量public String name ;public…

简易axure设计原型流程

我们这里做的是移动端的项目&#xff0c;基于提供的参考视频&#xff0c;截图了app的图片&#xff0c;作为实现原型时候的背景图。 原来实现的移动端的项目是基于vant开发的&#xff0c;因而下载了vant的元件库。参考地址: Vant 4 - A lightweight, customizable Vue UI libr…

python入门(八) 书体检测和识别

文章目录 实现思路:python环境开发工具传统形态学侵蚀&#xff0c;膨胀&#xff0c;张开&#xff0c;闭合侵蚀膨胀张开闭合 canny边缘检测算法检测书法文字案例实现步骤 本人工作中&#xff0c;用到了ai相关技术&#xff0c;但是java出身&#xff0c;所以从0开始学习&#xff0…

如何高效实现客户服务自助,打造在线产品帮助中心即可

随着互联网技术的不断发展&#xff0c;越来越多的企业开始重视客户服务自助和在线产品帮助中心的建设。这不仅可以提高客户满意度&#xff0c;还可以减轻客服人员的工作压力&#xff0c;降低企业的运营成本。本文将从以下几个方面探讨如何高效实现客户服务自助&#xff0c;打造…

干旱导致狮子捕猎家畜,马赛族人猎杀十余只,人与动物如何共存

最近&#xff0c;肯尼亚马赛族人在一场血腥的激战中&#xff0c;猎杀了数十头在野外流浪的狮子&#xff0c;对于这个事&#xff0c;大家也是议论纷纷。 肯尼亚旅游部长Peninah Malonza亲自乘坐直升机从内罗毕前来处理此事&#xff0c;足见问题的严重性。值得注意的是&#xff0…