使用 OpenCV 将 RGB 图像转换为 LMS,反之亦然

Posted

技术标签:

【中文标题】使用 OpenCV 将 RGB 图像转换为 LMS,反之亦然【英文标题】:Converting a RGB image to LMS, and vice versa, using OpenCV 【发布时间】:2019-05-18 16:48:18 【问题描述】:

我正在尝试使用 Python 中的 OpenCV 将图像从 RGB 转换为 LMS,反之亦然。据我了解,我应该使用给定的 3x3 变换矩阵并将其乘以 3x1 RGB/LMS 矩阵。使用的变换矩阵可以在here找到。

我已经在这个网站上探索了previouslyaskedquestions,但不幸的是他们使用的是 C++,这是一种我尚未精通的语言,我很难理解他们是如何解决他们的问题的.

到目前为止,这是我的代码:[已于 2019-05-19 解决]

import numpy as np
import cv2
#Transformation Matrix#
MsRGB  = np.zeros((3,3), dtype='float')
MHPE   = np.zeros((3,3), dtype='float')

MsRGB = np.array([[0.4124564, 0.3575761, 0.1804375],
                  [0.2126729, 0.7151522, 0.0721750],
                  [0.0193339, 0.1191920, 0.9503041]])

MHPE = np.array([[ 0.4002, 0.7076, -0.0808],
                 [-0.2263, 1.1653,  0.0457],
                 [      0,      0,  0.9182]])

Trgb2lms = MHPE @ MsRGB
Tlms2rgb = np.linalg.inv(Trgb2lms)

imgpath = "(insert file directory here)"
imgIN = cv2.imread(imgpath,cv2.IMREAD_UNCHANGED)
imgINrgb = cv2.cvtColor(imgIN, cv2.COLOR_BGR2RGB)

x,y,z = imgINrgb.shape
imgLMS = np.zeros((x,y,z), dtype='float')

imgReshaped = imgINrgb.transpose(2, 0, 1).reshape(3,-1)
imgLMS = Trgb2lms @ imgReshaped #Convert to LMS
imgOUT = Tlms2rgb @ imgLMS #Convert back to RGB

imgLMS = imgLMS.reshape(z, x, y).transpose(1, 2, 0).astype(np.uint8)
imgOUT = imgOUT.reshape(z, x, y).transpose(1, 2, 0).astype(np.uint8)

imgOUT = cv2.cvtColor(imgOUT, cv2.COLOR_RGB2BGR)
cv2.imshow('Input', imgIN)
cv2.imshow('LMS', imgLMS)
cv2.imshow('Output', imgOUT)
cv2.waitKey(0)
cv2.destroyAllWindows()

代码现在能够使用给定的变换矩阵对给定的 RGB 图像执行线性变换。结果可以在here找到。

【问题讨论】:

【参考方案1】:

鉴于您的问题的上下文,存在一些错误:

    T 未定义。从您的代码上下文来看,这应该是Trgb2lms,所以我们需要更改它们。

    从我可以从问题中收集到的信息,您正在对图像中的所有像素应用线性变换。为此,您需要重新调整矩阵的形状,以便我们有三行,其中每行对应于一个像素,然后是沿列的所有像素的解散。在这种情况下,reshape 方法不正确。您不仅需要打乱尺寸,使最后一个尺寸排在第一位,还需要设置reshape 的最后一个尺寸,使其为-1。这意味着我们将自动填充列,使其包含图像中的总像素数。

    最后,一旦你做了线性变换,你需要将矩阵重新整形为原始图像大小。您可以使用最终的reshape 调用,并使用原始调用中的x, yz 来推断图像尺寸。请记住,当我们重塑时,通道首先出现,因此我们必须再次排列尺寸。在我们进行转换后,您还需要返回无符号 8 位精度。

    同样为了比较,让我们通过逆变换来运行它,以确保我们拥有原始数据。

因此:

import numpy as np
import cv2
#Transformation Matrix#
MsRGB  = np.zeros((3,3), dtype='float')
MHPE   = np.zeros((3,3), dtype='float')

MsRGB = np.array([[0.4124564, 0.3575761, 0.1804375],
                  [0.2126729, 0.7151522, 0.0721750],
                  [0.0193339, 0.1191920, 0.9503041]])

MHPE = np.array([[ 0.4002, 0.7076, -0.0808],
                 [-0.2263, 1.1653,  0.0457],
                 [      0,      0,  0.9182]])

Trgb2lms = MHPE @ MsRGB

# Change
Tlms2rgb = np.linalg.inv(Trgb2lms)

imgpath = "(insert filename here)"
imgIN = cv2.imread(imgpath,cv2.IMREAD_UNCHANGED)
imgINrgb = cv2.cvtColor(imgIN, cv2.COLOR_BGR2RGB)

x,y,z = imgINrgb.shape
imgLMS = np.zeros((x,y,z), dtype='float')

#imgFlatten = imgINrgb.flatten()

# Change
imgReshaped = imgINrgb.transpose(2, 0, 1).reshape(3,-1)

# Change
imgLMS = Trgb2lms @ imgReshaped
imgOUT = Tlms2rgb @ imgLMS

# New
imgLMS = imgLMS.transpose(z, x, y).permute(1, 2, 0).astype(np.uint8)
imgOUT = imgOUT.transpose(z, x, y).permute(1, 2, 0).astype(np.uint8)

【讨论】:

感谢您抽出宝贵时间回复我的问题。我已经更新了代码并发布了结果。 @TidalSurge 那是因为我们需要将图像转换回 uint8。我也必须改变尺寸。给我一分钟。 @TidalSurge 好的。已编辑。立即尝试。 导致如下错误:imgReshaped = imgINrgb.permute(2, 0, 1).reshape(3,-1) AttributeError: 'numpy.ndarray' object has no attribute 'permute' @TidalSurge 抱歉让我修复。忘记了 permute 不是内置方法。

以上是关于使用 OpenCV 将 RGB 图像转换为 LMS,反之亦然的主要内容,如果未能解决你的问题,请参考以下文章

使用 OpenCV 从 android Camera2 将 YUV 转换为 RGB ImageReader 时出现问题,输出图像为灰度

将 RGB 图像转换为 3 个 HSI 输出

opencv python将图像读取为RGB

使用OpenCV从YUV颜色空间转换为RGB

如何使用 OpenCV 将 Bayer 转换为 RGB,反之亦然

在 OpenCV 中使用我自己的公式将 RGB 转换为灰色