UIImage cv::Mat 与 alpha 通道的转换

gun*_*yim 6 opencv objective-c uiimage ios

我使用以下代码进行相互UIImage*转换:cv::Mat

- (cv::Mat)cvMatFromUIImage:(UIImage *)image
{
  CGColorSpaceRef colorSpace = CGImageGetColorSpace(image.CGImage);
  CGFloat cols = image.size.width;
  CGFloat rows = image.size.height;

  cv::Mat cvMat(rows, cols, CV_8UC4); // 8 bits per component, 4 channels (color channels + alpha)

  CGContextRef contextRef = CGBitmapContextCreate(cvMat.data,                 // Pointer to  data
                                                 cols,                       // Width of bitmap
                                                 rows,                       // Height of bitmap
                                                 8,                          // Bits per component
                                                 cvMat.step[0],              // Bytes per row
                                                 colorSpace,                 // Colorspace
                                                 kCGImageAlphaNoneSkipLast |
                                                 kCGBitmapByteOrderDefault); // Bitmap info flags

  CGContextDrawImage(contextRef, CGRectMake(0, 0, cols, rows), image.CGImage);
  CGContextRelease(contextRef);

  return cvMat;
}
Run Code Online (Sandbox Code Playgroud)

-(UIImage *)UIImageFromCVMat:(cv::Mat)cvMat
{
  NSData *data = [NSData dataWithBytes:cvMat.data length:cvMat.elemSize()*cvMat.total()];
  CGColorSpaceRef colorSpace;

  if (cvMat.elemSize() == 1) {
      colorSpace = CGColorSpaceCreateDeviceGray();
  } else {
      colorSpace = CGColorSpaceCreateDeviceRGB();
  }

  CGDataProviderRef provider = CGDataProviderCreateWithCFData((__bridge CFDataRef)data);

  // Creating CGImage from cv::Mat
  CGImageRef imageRef = CGImageCreate(cvMat.cols,                                 //width
                                     cvMat.rows,                                 //height
                                     8,                                          //bits per component
                                     8 * cvMat.elemSize(),                       //bits per pixel
                                     cvMat.step[0],                            //bytesPerRow
                                     colorSpace,                                 //colorspace
                                     kCGImageAlphaNone|kCGBitmapByteOrderDefault,// bitmap info
                                     provider,                                   //CGDataProviderRef
                                     NULL,                                       //decode
                                     false,                                      //should interpolate
                                     kCGRenderingIntentDefault                   //intent
                                     );


  // Getting UIImage from CGImage
  UIImage *finalImage = [UIImage imageWithCGImage:imageRef];
  CGImageRelease(imageRef);
  CGDataProviderRelease(provider);
  CGColorSpaceRelease(colorSpace);

  return finalImage;
 }
Run Code Online (Sandbox Code Playgroud)

我从OpenCV 文档中获取了这些内容。我按如下方式使用它们:

UIImage *img = [UIImage imageNamed:@"transparent.png"];
UIImage *img2 = [self UIImageFromCVMat:[self cvMatFromUIImage:img]];
Run Code Online (Sandbox Code Playgroud)

然而这些函数会丢失 Alpha 通道信息。我知道这是因为标志kCGImageAlphaNonekCGImageAlphaNoneSkipLast,不幸的是我找不到通过更改这些标志来不丢失 alpha 信息的方法。

那么,如何在这两种类型之间相互转换而不丢失 alpha 信息呢?

这是我使用的图像:

在此输入图像描述

MoD*_*oDJ 1

您需要不通过kCGImageAlphaNoneSkipLast而是通过(kCGBitmapByteOrder32Host | kCGImageAlphaPremultipliedFirst)以获取 BGRA 格式的预乘 alpha。CoreGraphics 仅支持预乘 alpha。但是,您需要检查 OpenCV 如何表示像素中的 alpha,以确定如何告诉 OpenCV 像素已经被预乘。我使用的代码假设 OpenCV 是直接 alpha,所以你需要小心这一点。