Y = (0.257 * R) + (0.504 * G) + (0.098 * B) + 16
Cr = V = (0.439 * R) - (0.368 * G) - (0.071 * B) + 128
Cb = U = -(0.148 * R) - (0.291 * G) + (0.439 * B) + 128
然后我对矩阵进行了4:2:0色度子采样。我认为我做得正确,我从YUV矩阵中取2x2的子矩阵,将值从小到大排序,并取中间2个值的平均值。
接下来,我使用维基百科上的公式访问了Y、U和V平面:
size.total = size.width * size.height;
y = yuv[position.y * size.width + position.x];
u = yuv[(position.y / 2) * (size.width / 2) + (position.x / 2) + size.total];
v = yuv[(position.y / 2) * (size.width / 2) + (position.x / 2) + size.total + (size.total / 4)];
我正在使用OpenCV,所以我尽力最好地解释了这个:
y = src.data[(i*channels)+(j*step)];
u = src.data[(j%4)*step + ((i%2)*channels+1) + max];
v = src.data[(j%4)*step + ((i%2)*channels+2) + max + (max%4)];
src是YUV的子采样矩阵。我解释这个公式的方式正确吗?
以下是我将颜色还原为RGB的方法:
bgr.data[(i*channels)+(j*step)] = (1.164 * (y - 16)) + (2.018 * (u - 128)); // B
bgr.data[(i*channels+1)+(j*step)] = (1.164 * (y - 16)) - (0.813 * (v - 128)) - (0.391 * (u - 128)); // G
bgr.data[(i*channels+2)+(j*step)] = (1.164 * (y - 16)) + (1.596 * (v - 128)); // R
问题是我的图像没有恢复到原来的颜色。
以下是参考用的图片:http://i.stack.imgur.com/vQkpT.jpg(抽样)http://i.stack.imgur.com/Oucc5.jpg(输出)
我看到现在应该从YUV444转换为RGB,但我不太理解我在维基百科上找到的剪辑函数是做什么用的。
C = Y' − 16
D = U − 128
E = V − 128
R = clip(( 298 * C + 409 * E + 128) >> 8)
G = clip(( 298 * C - 100 * D - 208 * E + 128) >> 8)
B = clip(( 298 * C + 516 * D + 128) >> 8)
>>符号是否意味着我应该移位?
非常感谢您的帮助和建议!谢谢。
更新
尝试进行YUV444转换,但它只会使我的图像呈现绿色调。
y = src.data[(i*channels)+(j*step)];
u = src.data[(j%4)*step + ((i%2)*channels+1) + max];
v = src.data[(j%4)*step + ((i%2)*channels+2) + max + (max%4)];
c = y - 16;
d = u - 128;
e = v - 128;
bgr.data[(i*channels+2)+(j*step)] = clip((298*c + 409*e + 128)/256);
bgr.data[(i*channels+1)+(j*step)] = clip((298*c - 100*d - 208*e + 128)/256);
bgr.data[(i*channels)+(j*step)] = clip((298*c + 516*d + 128)/256);
以下是我的剪辑函数:
int clip(double value)
{
return (value > 255) ? 255 : (value < 0) ? 0 : value;
}
该函数的功能是将输入的值限制在0到255之间。