我的目标是将RGB像素转换为CIELab颜色空间,以进行一些只有在CIELab中才能实现的特殊计算。为此,我必须先将RGB转换为XYZ,这确实是非常困难的部分。
我尝试用Objective-C(主要使用了普通的C语言)来实现这个算法,但结果是错误的。
我的代码基于easyrgb.com提供的伪代码实现。他们有一个在线颜色转换器,工作得很好。他们说他们的伪代码与他们的转换器中使用的代码相同。
这是他们的伪代码:
然而,我使用相同的观察者和光源设置,但并未获得与他们工具相同的结果。
在我的测试中,我将这些值输入到他们的工具中,并获得了此XYZ结果,该结果与我的实现针对该RGB值产生的结果相差甚远。请参见屏幕截图:
我尝试用Objective-C(主要使用了普通的C语言)来实现这个算法,但结果是错误的。
我的代码基于easyrgb.com提供的伪代码实现。他们有一个在线颜色转换器,工作得很好。他们说他们的伪代码与他们的转换器中使用的代码相同。
这是他们的伪代码:
var_R = ( R / 255 ) //R from 0 to 255
var_G = ( G / 255 ) //G from 0 to 255
var_B = ( B / 255 ) //B from 0 to 255
if ( var_R > 0.04045 ) var_R = ( ( var_R + 0.055 ) / 1.055 ) ^ 2.4
else var_R = var_R / 12.92
if ( var_G > 0.04045 ) var_G = ( ( var_G + 0.055 ) / 1.055 ) ^ 2.4
else var_G = var_G / 12.92
if ( var_B > 0.04045 ) var_B = ( ( var_B + 0.055 ) / 1.055 ) ^ 2.4
else var_B = var_B / 12.92
var_R = var_R * 100
var_G = var_G * 100
var_B = var_B * 100
//Observer. = 2°, Illuminant = D65
X = var_R * 0.4124 + var_G * 0.3576 + var_B * 0.1805
Y = var_R * 0.2126 + var_G * 0.7152 + var_B * 0.0722
Z = var_R * 0.0193 + var_G * 0.1192 + var_B * 0.9505
这是我尝试用Objective-C / C实现它的方法:
void convertRGBtoXYZ(NSInteger * inR, NSInteger * inG, NSInteger * inB, CGFloat * outX, CGFloat * outY, CGFloat * outZ) {
// http://www.easyrgb.com/index.php?X=MATH&H=02#text2
CGFloat var_R = (*inR / 255); //R from 0 to 255
CGFloat var_G = (*inG / 255); //G from 0 to 255
CGFloat var_B = (*inB / 255); //B from 0 to 255
if (var_R > 0.04045f) {
var_R = powf(( (var_R + 0.055f) / 1.055f), 2.4f);
} else {
var_R = var_R / 12.92f;
}
if (var_G > 0.04045) {
var_G = powf(( (var_G + 0.055f) / 1.055f), 2.4f);
} else {
var_G = var_G / 12.92f;
}
if (var_B > 0.04045f) {
var_B = powf(( (var_B + 0.055f) / 1.055f), 2.4f);
} else {
var_B = var_B / 12.92f;
}
var_R = var_R * 100;
var_G = var_G * 100;
var_B = var_B * 100;
//Observer. = 2°, Illuminant = D65
*outX = var_R * 0.4124f + var_G * 0.3576f + var_B * 0.1805f;
*outY = var_R * 0.2126f + var_G * 0.7152f + var_B * 0.0722f;
*outZ = var_R * 0.0193f + var_G * 0.1192f + var_B * 0.9505f;
}
然而,我使用相同的观察者和光源设置,但并未获得与他们工具相同的结果。
在我的测试中,我将这些值输入到他们的工具中,并获得了此XYZ结果,该结果与我的实现针对该RGB值产生的结果相差甚远。请参见屏幕截图:
得到的Lab颜色值与Photoshop告诉我的相当接近,因此转换器非常有效。
然而,上面的C代码给我这个结果:
X = 35.76... // should be 42.282
Y = 71.52... // should be 74.129
Z = 11.92... // should be 46.262
你知道这个失败的原因吗?是我的实现有误,还是我需要其他常量?
如果你知道一些经过测试的RGB到XYZ、XYZ到CIELab或RGB到CIELab、XYZ到Lab或RGB到Lab的实现,请不要犹豫在这里发布。
基本上,我想做的就是计算两种颜色之间的偏差,也称为Delta-E。这就是为什么我需要从RGB转换到XYZ再转换到Lab(或CIELab)的原因……