1

最近邻缩放有效:当我使用TYPE_NEAREST_NEIGHBOR.

尽管它是 Scala 代码,但所有使用的库都是标准 Java 库。

职能:

def getBufferedImage(imageFile: java.io.File): BufferedImage = {
    ImageIO.read(imageFile)
}

def scaleImage(image: BufferedImage, minSize: Double): BufferedImage = {
    val before: BufferedImage = image
    val w = before.getWidth()
    val h = before.getHeight()
    val affit = new AffineTransform()
    var scale = 1.0
    if(h < w) {
      if(h > 0) {
        scale = minSize / h
      }
    } else {
      if(w > 0) {
        scale = minSize / w
      }
    }
    affit.scale(scale, scale)
    val affitop = new AffineTransformOp(affit, AffineTransformOp.TYPE_BICUBIC)
    affitop.filter(before, null)
}

def getImageJpegByteArray(image: BufferedImage): Array[Byte] = {
    val baos = new java.io.ByteArrayOutputStream()
    val mcios = new MemoryCacheImageOutputStream(baos)
    ImageIO.write(image, "jpeg", mcios)
    mcios.close()
    baos.toByteArray
}

调用代码片段:

val img = getBufferedImage(imageFile)
val scaledImg = scaleImage(img, 512)
val result = getImageJpegByteArray(scaledImg)
// result is written to SQLite database

result被写入 SQLite 数据库。如果我从数据库中下载它并将其保存为 JPEG 文件,则生成的 JPEG 是

  • 正如预期的那样,如果我使用AffineTransformOp.TYPE_NEAREST_NEIGHBOR
  • 如果我使用完全黑AffineTransformOp.TYPE_BILINEAR
  • 如果我使用完全黑AffineTransformOp.TYPE_BICUBIC

因此,我指责AffineTransformOp是错误的...... 我该如何解决这个问题?

文件的幻数result总是ff d8 ff与 JPEG 的预期一致。

细节

Java 版本:Java HotSpot(TM) 64 位服务器 VM,Java 1.7.0_71

操作系统:苹果、OS X 10.9.5

测试图片:http ://www.photos-public-domain.com/wp-content/uploads/2012/05/thundercloud-plum-blossoms.jpg

4

1 回答 1

1

我能够在 OS X 10.10.4 上的 Java 1.7.0_71 上重现您的问题(我用 Java 重写了您的代码,如果您有兴趣,我可以发布完整的代码)。

无论如何,问题在于AffineTransformOp它本身就是错误的。在我的测试程序中,我使用最小的 Swing 显示了图像JFrame,缩放后的图像在那里看起来都很好。这可能是评论中大多数人不理解这个问题的原因。

部分问题是,BufferedImageAffineTransformOp您没有为filter方法提供目的地(在您的情况下是第二个参数null)时,它会为您创建一个。此图像将获得 type BufferedImage.TYPE_INT_ARGB。这是来自AffineTransformOp.createCompatibleDestImage()(第 456-468 行,我保留了格式,以便更容易发现)的相关代码:

ColorModel cm = src.getColorModel();
if (interpolationType != TYPE_NEAREST_NEIGHBOR &&
    (cm instanceof IndexColorModel ||
     cm.getTransparency() == Transparency.OPAQUE)
{
    image = new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB);
}
else {
    image = new BufferedImage(cm,
              src.getRaster().createCompatibleWritableRaster(w,h),
              cm.isAlphaPremultiplied(), null);
}

注意 的特殊情况TYPE_NEAREST_NEIGHBOR,它解释了为什么在使用最近邻算法时会出现不同的行为。但是,通常这一切都很好(正如我所说,图像在 Swing 组件中显示得很好)。

当您尝试将此图像存储为 JPEG 时,就会出现问题。这些年来,存在很多与 ImageIO JPEG 插件相关的困惑和问题,以及它是否允许您使用 Alpha 通道编写图像(如您的TYPE_INT_ARGB图像)。它确实允许这样做。但是,大多数情况下 ARGB JPEG 会被误解为 CMYK JPEG(因为它们是 4 个通道,并且将 ARGB 数据存储在 JPEG 中非常奇特),并且会以所有时髦的颜色显示。不过,在你的情况下,它似乎全是黑色的......

因此,有两种可能的解决方案:

  • 以支持 Alpha 通道的文件格式编写图像,例如 PNG 或 TIFF(TIFF 需要额外的插件,因此它可能不是最佳选择)。像这样:

    ImageIO.write(image, "PNG", mcios);
    
  • 或者,在存储为 JPEG 之前,请确保您BufferedImage是没有 Alpha 通道的像素格式。您可以在缩放后执行此操作,但最简单(也是最快)的方法是仅提供AffineTransformOp明确的目标图像,如下所示:

    Rectangle newSize = affitop.getBounds2D(before).getBounds();
    return affitop.filter(before, 
          new BufferedImage(newSize.width,  newSize.height, BufferedImage.TYPE_3BYTE_BGR));
    

这是您的图像,由程序缩放,使用 JPEG 格式和TYPE_3BYTE_BGR

缩放图像

我确信您可以将我的 Java 代码重写回 Scala。:-)

于 2015-08-01T10:09:00.647 回答