• WPF 已知问题 BitmapDecoder.Create 不支持传入 Asynchronous 的文件流


    这是在 GitHub 上有小伙伴报的问题,在 WPF 中,不支持调用 BitmapDecoder.Create 方法,传入的 FileStream 是配置了 FileOptions.Asynchronous 选项的文件流。本质原因是 WIC 层不支持,和 WPF 没有关系

    GitHub 链接: BitmapDecoder.Create does not handle FileStream with FileOptions.Asynchronous · Issue #4355 · dotnet/wpf

    现象是传入 BitmapDecoder.Create 的 FileStream 配置了 FileOptions.Asynchronous 选项,代码如下

    using var fs = new FileStream("image.jpg",
    	FileMode.Open,
    	FileAccess.Read,
    	FileShare.Read,
    	4096,
    	FileOptions.Asynchronous);
    
    var decoder = BitmapDecoder.Create(fs, BitmapCreateOptions.PreservePixelFormat, BitmapCacheOption.None);
    

    运行以上代码将会抛出 ArgumentException 而创建解码器失败

    本质原因是 WIC 层不支持 FileStream 配置了 FileOptions.Asynchronous 选项。在 BitmapDecoder.Create 的底层,调用了 IWICImagingFactory_CreateDecoderFromFileHandle_Proxy function - Win32 apps 方法创建解码器,代码如下

            public static BitmapDecoder Create(
                Stream bitmapStream,
                BitmapCreateOptions createOptions,
                BitmapCacheOption cacheOption
                )
            {
                // 忽略代码
                return CreateFromUriOrStream(
                    null,
                    null,
                    bitmapStream,
                    createOptions,
                    cacheOption,
                    null,
                    true
                    );
            }
    
            internal static BitmapDecoder CreateFromUriOrStream(
                Uri baseUri,
                Uri uri,
                Stream stream,
                BitmapCreateOptions createOptions,
                BitmapCacheOption cacheOption,
                RequestCachePolicy uriCachePolicy,
                bool insertInDecoderCache
                )
            {
                // 忽略代码
                    decoderHandle = BitmapDecoder.SetupDecoderFromUriOrStream(
                        finalUri,
                        stream,
                        cacheOption,
                        out clsId,
                        out isOriginalWritable,
                        out uriStream,
                        out unmanagedMemoryStream,
                        out safeFilehandle
                        );
                // 忽略代码
            }
    
            internal static SafeMILHandle SetupDecoderFromUriOrStream(
                Uri uri,
                Stream stream,
                BitmapCacheOption cacheOption,
                out Guid clsId,
                out bool isOriginalWritable,
                out Stream uriStream,
                out UnmanagedMemoryStream unmanagedMemoryStream,
                out SafeFileHandle safeFilehandle
                )
            {
                        using (FactoryMaker myFactory = new FactoryMaker())
                        {
                            HRESULT.Check(UnsafeNativeMethods.WICImagingFactory.CreateDecoderFromFileHandle(
                                myFactory.ImagingFactoryPtr,
                                safeFilehandle,
                                ref vendorMicrosoft,
                                metadataFlags,
                                out decoder
                                ));
                        }
            }
    

    也就是说最底层调用就是通过 CreateDecoderFromFileHandle 方法创建解码器,而 CreateDecoderFromFileHandle 的定义如下

            internal static class WICImagingFactory
            {
                [DllImport(DllImport.WindowsCodecs, EntryPoint = "IWICImagingFactory_CreateDecoderFromFileHandle_Proxy")]
                internal static extern int /*HRESULT*/ CreateDecoderFromFileHandle(
                    IntPtr pICodecFactory,
                    Microsoft.Win32.SafeHandles.SafeFileHandle  /*ULONG_PTR*/ hFileHandle,
                    ref Guid guidVendor,
                    UInt32 metadataFlags,
                    out IntPtr /* IWICBitmapDecoder */ ppIDecode);
            }
    

    从以上代码可以看到是在 WindowsCodecs.dll 也就是 WIC 层的 IWICImagingFactory_CreateDecoderFromFileHandle_Proxy 抛出失败的

    在 FileStream 创建中,如果传入了 FileOptions.Asynchronous 参数,将会在 Windows 下,调用的是 CreateFileW function (fileapi.h) - Win32 apps 方法,在这个方法里面将会设置 FILE_FLAG_OVERLAPPED 参数。不过我没有从网上找到在设置了 FILE_FLAG_OVERLAPPED 参数的文件句柄将在 IWICImagingFactory_CreateDecoderFromFileHandle_ProxyIWICImagingFactory::CreateDecoderFromFileHandle 方法不支持的知识

    我写了一个简单的 demo 程序,用来测试是否 FileOptions.Asynchronous 参数的文件句柄将会在 WIC 层不支持

        class Program
        {
            static void Main(string[] args)
            {
                CheckHResult(UnsafeNativeMethods.WICCodec.CreateImagingFactory(UnsafeNativeMethods.WICCodec.WINCODEC_SDK_VERSION,
                    out var pImagingFactory));
    
                using var fs = new FileStream("image.jpg",
                    FileMode.Open,
                    FileAccess.Read,
                    FileShare.Read,
                    4096,
                    FileOptions.Asynchronous);
    
                Guid vendorMicrosoft = new Guid(MILGuidData.GUID_VendorMicrosoft);
                UInt32 metadataFlags = (uint)WICMetadataCacheOptions.WICMetadataCacheOnDemand;
    
                CheckHResult
                (
                    UnsafeNativeMethods.WICImagingFactory.CreateDecoderFromFileHandle
                    (
                        pImagingFactory,
                        fs.SafeFileHandle,
                        ref vendorMicrosoft,
                        metadataFlags,
                        out var decoder
                    )
                );
            }
    
            static void CheckHResult(int hr)
            {
                if (hr < 0)
                {
                    Exception exceptionForHR = Marshal.GetExceptionForHR(hr, (IntPtr)(-1));
    
                    throw exceptionForHR;
                }
            }
        }
    

    通过以上代码可以看到,如果 FileStream 的创建参数里面加上了 FileOptions.Asynchronous 那么将会在 CreateDecoderFromFileHandle 抛出错误

    因此在 WPF 中,调用 BitmapDecoder.Create 方法,传入的带 FileOptions.Asynchronous 的 FileStream 抛出错误,不是 WPF 层的锅,而是 WIC 层不支持。在 GitHub 上报告的作者 Nikita Kazmin 给了一个我同意的建议是 WPF 在 BitmapDecoder.Create 方法里面应该判断一下,如果传入的 FileStream 是异步的,那么在 WPF 层抛出错误,这样方便开发者了解不能这样使用

    我也有另一个想法,如果是 FileStream 是异步的,不如完全读取到内存里面,这样开发者也就可以不关注这部分的逻辑。我当前将此逻辑放入到 WPF 仓库中,详细请看 https://github.com/dotnet/wpf/pull/4966

    本文所有代码放在 githubgitee 欢迎小伙伴访问

    当前的 WPF 在 https://github.com/dotnet/wpf 完全开源,使用友好的 MIT 协议,意味着允许任何人任何组织和企业任意处置,包括使用,复制,修改,合并,发表,分发,再授权,或者销售。在仓库里面包含了完全的构建逻辑,只需要本地的网络足够好(因为需要下载一堆构建工具),即可进行本地构建

    博客园博客只做备份,博客发布就不再更新,如果想看最新博客,请到 https://blog.lindexi.com/

    知识共享许可协议
    本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。欢迎转载、使用、重新发布,但务必保留文章署名[林德熙](http://blog.csdn.net/lindexi_gd)(包含链接:http://blog.csdn.net/lindexi_gd ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我[联系](mailto:lindexi_gd@163.com)。
  • 相关阅读:
    [转]HD钱包的助记词与密钥生成原理
    [转]简单科普私钥、地址、助记词、Keystore的区别
    [转]Sequelize 中文API文档-4. 查询与原始查询
    [转]Node.JS使用Sequelize操作MySQL
    [转]OmniLayer / omnicore API 中文版
    [转]usdt omnicore testnet 测试网络
    [转]USDT与omniCore钱包
    [转]BTC RPC API GetTransaction
    [转]比特币测试链——Testnet介绍
    [转]BTC手续费计算,如何设置手续费
  • 原文地址:https://www.cnblogs.com/lindexi/p/15131064.html
Copyright © 2020-2023  润新知