将 RGB2CMYK 方法从 c++ 移植到 java 时出现问题

2022-09-03 14:56:21

我正在尝试将方法从c ++转换为java。方法是这样的方法:

void rgb2cmyk(cv::Mat& src, std::vector<cv::Mat>& cmyk)
{
    CV_Assert(src.type() == CV_8UC3);

    cmyk.clear();
    for (int i = 0; i < 4; ++i)
       cmyk.push_back(cv::Mat(src.size(), CV_32F));

    for (int i = 0; i < src.rows; ++i)
    {
        for (int j = 0; j < src.cols; ++j)
        {
            cv::Vec3b p = src.at<cv::Vec3b>(i,j);

            float r = p[2] / 255.;
            float g = p[1] / 255.;
            float b = p[0] / 255.;
            float k = (1 - std::max(std::max(r,g),b));

            cmyk[0].at<float>(i,j) = (1 - r - k) / (1 - k); 
            cmyk[1].at<float>(i,j) = (1 - g - k) / (1 - k);
            cmyk[2].at<float>(i,j) = (1 - b - k) / (1 - k);
            cmyk[3].at<float>(i,j) = k;
        }
    }
}

问题在于OpenCv的方法。以下是一些详细信息:

  1. 我没有在java中找到方法。不知道任何替代者。CV_Assert
  2. cmyk.push_back替换为cmyk[i].pushback
  3. 我已经用了替换,它是显示没有错误Matcv::Vec3b
  4. std::max替换为Math.max
  5. 问题是分配给cmyk[0].at<float>(i,j)

有没有人有建议或任何更好的方法将此方法更改为java。

提前感谢您的帮助...

编辑

我做了什么

public void rgb2xmyk(Mat src,Mat[] cmyk)
{
    //CV_Assert(src.type() == CV_8UC3);
    //cmyk.clear();
    for (int i = 0; i < 4; ++i)
        cmyk[i].push_back(new Mat(src.size(), CvType.CV_32F));

    for (int i = 0; i < src.rows; ++i)
    {
         for (int j = 0; j < src.cols; ++j)
         {
             double[] p = src.get(i,j);
             float r = (float) (p[2] / 255.);
             float g = (float) (p[1] / 255.);
             float b = (float) (p[0] / 255.);
             float k = (1 - Math.max(Math.max(r,g),b));

             cmyk[0].at<float>(i,j) = (1 - r - k) / (1 - k); 
             cmyk[1].at<float>(i,j) = (1 - g - k) / (1 - k);
             cmyk[2].at<float>(i,j) = (1 - b - k) / (1 - k);
             cmyk[3].at<float>(i,j) = k;
        }
    }
}

答案 1

您必须确保 Mat 的 cmyk 数组的大小为 4。在 for 循环中,我建议你使用 setTo:

for (int i = 0; i < 4; ++i)
    cmyk[i].setTo(new Mat(src.size(), CvType.CV_32F));

在嵌套的循环中,你填充你的cmyk,我会使用放置方法

cmyk[0].put(i,j,new Float[] {(1 - r - k) / (1 - k)});
cmyk[1].put(i,j,new Float[] {(1 - g - k) / (1 - k)});
cmyk[2].put(i,j,new Float[] {(1 - b - k) / (1 - k)});
cmyk[3].put(i,j,new Float[] {k});

...希望这有帮助


答案 2

您可能会发现在此行中:

cmyk[0].at<float>(i,j) = (1 - r - k) / (1 - k); 

在转换为浮点型之前,将结果作为 int 进行处理,因为每个块都有整数。尝试:

cmyk[0].at<float>(i,j) = (1.0f - r - k) / (1.0f - k);

我认为您已经对如何访问该数组中的元素有了一个明智的答案(通过.put)。


推荐