.NET解所有相机RAW格式照片

再聊.NET解相机RAW格式照片

上次我发了一篇文章《用.NET解索尼相机ARW格式照片》,提到通过安装Sony Raw File Decoder的方式,然后调用Windows Imaging Components来解析RAW格式文件。后来我经过进一步研究、探索,发现还有更简单的办法。

新的方法实在是太简单、好用了,相比之下,我前一篇文章简直就是在“挖坑”。

其实啥都不装,什么相机都支持!

其实Windows 10自带了一个RAW格式解码器,也集成在Windows Imaging Components中,通过SharpDX.Direct2D1的几行代码,可以将这个解码器的信息调出来:

// 安装NuGet包:SharpDX.Direct2D1
using var wic = new ImagingFactory2();
using var decoder = new BitmapDecoder(wic, file, DecodeOptions.CacheOnDemand);
string json = JsonSerializer.Serialize(decoder.DecoderInfo, new JsonSerializerOptions { WriteIndented = true });
Console.WriteLine(json);

运行结果如下(为突出重点,有少量删减):

{
  "PixelFormats": [
    "6fddc324-4e03-4bfe-b185-3d77768dc90d"
  ],
  "ColorManagementVersion": "1.0.0.0\u0000",
  "MimeTypes": "image/3FR,image/ARI,image/ARW,image/BAY,image/CAP,image/CR2,image/CR3,image/CRW,image/DCS,image/DCR,image/DRF,image/EIP,image/ERF,image/FFF,image/IIQ,image/K25,image/KDC,image/MEF,image/MOS,image/MRW,image/NEF,image/NRW,image/ORF,image/ORI,image/PEF,image/PTX,image/PXN,image/RAF,image/RAW,image/RW2,image/RWL,image/SR2,image/SRF,image/SRW,image/X3F,image/DNG\u0000",
  "FileExtensions": ".3FR,.ARI,.ARW,.BAY,.CAP,.CR2,.CR3,.CRW,.DCS,.DCR,.DRF,.EIP,.ERF,.FFF,.IIQ,.K25,.KDC,.MEF,.MOS,.MRW,.NEF,.NRW,.ORF,.ORI,.PEF,.PTX,.PXN,.RAF,.RAW,.RW2,.RWL,.SR2,.SRF,.SRW,.X3F,.DNG\u0000",
  "ContainerFormat": "fe99ce60-f19c-433c-a3ae-00acefa9ca21",
  "IsAnimationSupported": false,
  "IsChromakeySupported": false,
  "IsLosslessSupported": true,
  "IsMultiframeSupported": false,
  "Author": "Microsoft Corporation\u0000",
  "Version": "10.0.18362.1\u0000",
  "SpecVersion": "1.0.0.0\u0000",
  "FriendlyName": "Microsoft Raw Image Decoder\u0000",
  "ComponentType": 1,
  "CLSID": "41945702-8302-44a6-9445-ac98e8afa086",
  "SigningStatus": 1,
  "VendorGUID": "f0e749ca-edef-4589-a73a-ee0e626a2a2b",
}

可见,什么都不用装,就已经支持了高达36种RAW格式文件,索尼的.ARW、佳能的.CR2和.CR3和尼康的.NEF都在列——就可能就是为什么Windows 10可以直接打开相机的RAW格式文件。

使用上次文章中的同样代码,即可将.ARW格式文件转换为jpeg:

// 依赖于WPF,不用装NuGet包
var decoder = BitmapDecoder.Create(new Uri(@"DSC05458.ARW"), BitmapCreateOptions.None, BitmapCacheOption.Default);
var transformedBitmap = new TransformedBitmap(decoder.Frames[0], Transform.Identity);
var jpg = new JpegBitmapEncoder();
jpg.Frames.Add(BitmapFrame.Create(transformedBitmap));
using var stream = new MemoryStream();
jpg.Save(stream);

上次还挖了个坑说如果是我,一般会选择用SharpDX而不是WPF,但我又没说SharpDX的代码该怎么写,这里面我将SharpDX的代码贴出来(运行效果完全一样):

// 安装NuGet包:SharpDX.Direct2D1
using var wic = new ImagingFactory2();
using FormatConverter converter = LoadImage(wic, @"DSC00115.ARW");
Util.Image(SaveToJpeg(wic, converter)).Dump();

static byte[] SaveToJpeg(ImagingFactory2 wic, BitmapSource source)
{
    using var ms = new MemoryStream();
    using (var encoder = new JpegBitmapEncoder(wic, ms))
    {
        using (var frame = new BitmapFrameEncode(encoder))
        {
            frame.Options.ImageQuality = 0.7f;
            frame.Initialize();
            frame.WriteSource(source);
            frame.Commit();
        }
        encoder.Commit();
    }
    return ms.ToArray();
}

static FormatConverter LoadImage(ImagingFactory2 wic, string file)
{
    using var decoder = new BitmapDecoder(wic, file, DecodeOptions.CacheOnDemand);
    decoder.Dump();
    var converter = new FormatConverter(wic);
    converter.Initialize(decoder.GetFrame(0), PixelFormat.Format32bppPBGRA);
    return converter;
}

代码中我加入了缩放,运行上次的.ARW文件后,可以得出一样的jpeg图片。

跨平台?没问题!

故名思义Windows Imaging Componnets,显然只有Windows上才能运行。想跨平台读取相机RAW格式文件就必须另找一个库——Magick.NET,其使用也非常简单,甚至比WIC更简单,只要两行代码!:

// 安装NuGet包:Magick.NET-Q8-AnyCPU
using var image = new MagickImage(@"DSC00115.ARW");
byte[] bytes = image.ToByteArray(MagickFormat.Jpeg);

除了读取保存,Magick.NET还能缩放图片、转换pdf、加水印、读取Exif数据、无损压缩、绘图等功能,具体功能可以参见:https://github.com/dlemstra/Magick.NET/blob/master/docs/Readme.md。

另外,Magick.NET还支持超过200种图片格式,其中甚至还包括Photoshop的psd文件。可以在这个链接中查看是否支持你所需要的格式:https://imagemagick.org/index.php

……更别它还可以跨平台。

有一点需要注意,它的NuGet包有许多个,初一看可能会一惊: .NET解所有相机RAW格式照片

这里可以说一下,首先它有Q8、Q16和Q16-HDRI三种版本:

  • Q8表示一个像素使用8位颜色深度,它占用内存最小;
  • Q16表示一个像素使用16位颜色深度,比Q8多一倍;
  • Q16-HDRI则使用32位浮点型来表示颜色,比Q16再多一倍;

正常使用建议Q8即可,有微单/单反修图需求的,可以考虑Q16/Q16-HDRI。

另外它还有x86、x64和AnyCPU三个平台版本,一般选AnyCPU,但注意Magick.NET是基于本地代码,因此它需要下载多个平台,因此AnyCPU大小(45.34MB)比x64(20.85MB)大一倍。

缺点?

有这么多优点,还要什么自行车?……其实它也是有缺点的,有个明显的缺点,就是性能慢。我测试了上次那张ARW格式文件,性能分析如下(平均需要3.7秒):

次数 分配内存 内存提高 耗时
1 43,918,792 1,080 3739
2 43,884,944 160 3748
3 43,966,000 664 3878
4 44,015,928 80 3778
5 43,902,784 416 3747

换成WIC,性能数据如下,耗时只要705毫秒,明显快得多:

次数 分配内存 内存提高 耗时
1 11,939,072 1,488 713
2 11,939,768 10,256 709
3 12,013,872 4,608 705
4 11,931,664 96 705
5 11,919,384 -25,968 704

总结

如果你用Windows 10,则什么都不用装,就能体验到极致性能的RAW解析工具,因为系统自带了Microsoft Raw Image Decoder。

如果你想跨平台(但不特别在意性能),则可以使用开源的Magick.NET,它提供最便利的API和最省心的功能、格式支持,特别强大。但话又说回来,RAW这种东西一般都是骚操作才需要,谁在意跨平台呢?

我也特意试了一下收费的Aspose.Imaging,但解析RAW格式文件不是它的长项,直接不支持。

本想研究一下libraw,但它只提供了C API——也不是不能用。正准备用P/Invoke时刚正面就有了本文中的发现。


上一篇:Android Studio调用Github中的json文件


下一篇:Nginx编译配置脚本篇(6)- 系统环境相关脚本