本文告诉大家如何使用 Vortice 进行 D2D 的离屏渲染功能,本文将在一个纯控制台无窗口的应用下,使用 Direct2D1 进行离屏绘制,将绘制结果保存为本地图片文件

本文属于使用 Vortice 调用 DirectX 系列博客,也属于 DirectX 系列博客,本文属于入门级博客,但在阅读本文之前,期望大家了解了 DirectX 的基础概念

本文使用的 Vortice 是 SharpDx 的代替品,是对 DirectX 的底层 C# 封装。使用 Vortice 底层库,能让 C# 代码比较方便的和 DirectX 对接。尽管本文使用的是 Vortice 库来调用 DirectX 相关的接口,但不代表着只有 Vortice 库能做此实现,可以将 Vortice 换成其他的对 DirectX 封装的库,例如 SharpDx 或者是 Silk.NET 等,更换之后只是调用的方法或者是参数等稍微有点不相同,但是实现思路都是相同的

使用 Direct2D1 离屏渲染技术,可以进行脱离具体的窗口调用渲染,可以不需要占用主线程的时间,采用后台线程驱动执行渲染。可以实现在某些性能敏感的业务上,预先准备好渲染内容,从而提升性能等

新建一个 dotnet 6 的控制台项目,咱将在此新建的项目里面完成一个简单的 Direct2D1 离屏渲染的控制台应用。本文不会贴所有的代码,如果按照本文给出的代码构建不通过或者遇到其他问题,还请到本文末尾获取所有源代码,将源代码拉到本地构建

按照惯例,在开始之前,先通过 NuGet 安装必要的库。在 dotnet 6 项目里,采用 SDK 风格的 csproj 项目文件格式,可以通过编辑项目文件的方式快速安装 NuGet 库。可以在 VisualStudio 里,右击项目,点击编辑项目文件,或者双击项目都可以,替换项目文件为以下代码即可完成安装

<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<PackageReference Include="Vortice.Direct2D1" Version="2.1.32" />
<PackageReference Include="Vortice.Win32" Version="1.6.2" />
</ItemGroup>
</Project>

以上代码安装了 Vortice.Direct2D1 库用来对接 Direct2D1 的逻辑,安装了 Vortice.Win32 用来辅助处理杂项逻辑

打开 Program.cs 文件,开始编写离线渲染逻辑。开始编写代码之前,先引用命名空间

using Vortice.Mathematics;
using Vortice.WIC;
using D2D = Vortice.Direct2D1;
using PixelFormat = Vortice.DCommon.PixelFormat;

大家都知道,使用 D2D 时,最重要的就是获取到 ID2D1RenderTarget 用来作为绘制的画布。创建 ID2D1RenderTarget 的方法有很多个,本文是通过 ID2D1Factory1 工厂调用 CreateWicBitmapRenderTarget 方法,从一个 IWICBitmap 创建的

也就是说想要获取到 ID2D1RenderTarget 进行绘制,就需要能先拿到 IWICBitmap 类型的对象。创建 IWICBitmap 类型的对象需要通过 WIC 工厂进行创建。于是先创建工厂

        using var wicImagingFactory = new IWICImagingFactory();

通过 IWICImagingFactory 工厂创建 IWICBitmap 对象,需要调用 CreateBitmap 方法,传入尺寸和颜色格式。颜色格式里面只有一些是 D2D 支持的,本文这里采用常用的 PixelFormat32bppPBGRA 格式

        using IWICBitmap wicBitmap =
wicImagingFactory.CreateBitmap(1000, 1000, Win32.Graphics.Imaging.Apis.GUID_WICPixelFormat32bppPBGRA);

获取到 IWICBitmap 类型的对象,即可开始通过 D2D 工厂创建 ID2D1RenderTarget 画布。先创建 D2D 工厂

        using D2D.ID2D1Factory1 d2DFactory = D2D.D2D1.D2D1CreateFactory<D2D.ID2D1Factory1>();

接着设置渲染参数

        var renderTargetProperties = new D2D.RenderTargetProperties(PixelFormat.Premultiplied);

创建 ID2D1RenderTarget 对象

        D2D.ID2D1RenderTarget d2D1RenderTarget =
d2DFactory.CreateWicBitmapRenderTarget(wicBitmap, renderTargetProperties);

以上就是最核心的步骤了,获取到 ID2D1RenderTarget 对象,即可开始 D2D 的绘制逻辑,如下面代码,修改画布颜色

        using var renderTarget = d2D1RenderTarget;
// 开始绘制逻辑
renderTarget.BeginDraw(); // 随意创建颜色
var color = new Color4((byte) Random.Shared.Next(255), (byte) Random.Shared.Next(255),
(byte) Random.Shared.Next(255));
renderTarget.Clear(color); renderTarget.EndDraw();

如此即可将内容绘制到 IWICBitmap 上

接下来是将 IWICBitmap 的内容保存到本地的图片,保存 IWICBitmap 需要先对 IWICBitmap 进行编码,编码时需要使用 WIC 工厂创建编码器,接着传入编码的格式和编码的输出

先打开一个文件用来存放编码的输出

        var file = @"D2D.png";
using (var fileStream = File.OpenWrite(file))
{
// 忽略代码
}

通过 WIC 工厂创建编码器,设置编码的格式是 png 格式

            using var wicBitmapEncoder =
wicImagingFactory.CreateEncoder(Win32.Graphics.Imaging.Apis.GUID_ContainerFormatPng);

设置编码的输出到文件

            wicBitmapEncoder.Initialize(fileStream);

从编码器创建出一张图片

            using var wicFrameEncode = wicBitmapEncoder.CreateNewFrame(out var _);
wicFrameEncode.Initialize();

将上文完成绘制的 IWICBitmap 输入到编码器里面

            wicFrameEncode.WriteSource(wicBitmap);

完成逻辑之后提交一下

            wicFrameEncode.Commit();
wicBitmapEncoder.Commit();

如此执行完成,即可将绘制的内容保存到本地文件里。这就是本文的采用 D2D 进行离屏绘制的方法

想不开的话,可以测试一下调用渲染时是否能跑满 GPU 资源,稍微更改一下渲染的代码,从原本的调用 Clear 修改颜色,修改为以下逻辑

        using var renderTarget = d2D1RenderTarget;
var stopwatch = Stopwatch.StartNew();
while (true)
{
// 开始绘制逻辑
renderTarget.BeginDraw(); // 随意创建颜色
var color = new Color4((byte) Random.Shared.Next(255), (byte) Random.Shared.Next(255),
(byte) Random.Shared.Next(255));
renderTarget.Clear(color);
color = new Color4(GetRandom(), GetRandom(), GetRandom());
using var brush = renderTarget.CreateSolidColorBrush(color); for (int i = 0; i < 1000; i++)
{
renderTarget.DrawEllipse(new D2D.Ellipse(new Vector2(GetRandom(), GetRandom()), 5, 5), brush, 2);
} stopwatch.Stop();
Console.WriteLine($"Draw: {stopwatch.ElapsedMilliseconds}");
stopwatch.Restart(); renderTarget.EndDraw(); stopwatch.Stop();
Console.WriteLine($"EndDraw: {stopwatch.ElapsedMilliseconds}");
stopwatch.Restart(); byte GetRandom() => (byte) Random.Shared.Next(255);
}

尝试运行代码,看看任务管理器里面,显示当前进程是否有用到 GPU 资源,以及占用了多少 GPU 资源

本文的代码放在githubgitee 欢迎访问

可以通过如下方式获取本文的源代码,先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin bb1f1f3db2cf7317341e830d1e3adb14df67a71e

以上使用的是 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin bb1f1f3db2cf7317341e830d1e3adb14df67a71e

获取代码之后,进入 WakolerwhaKanicabirem 文件夹

渲染部分,关于 SharpDx 使用,包括入门级教程,请参阅:

在 WPF 框架的渲染部分,请参阅: WPF 底层渲染_lindexi_gd的博客-CSDN博客

更多关于我博客请参阅 博客导航

交流 Vortice 技术,欢迎加群: 622808968

dotnet C# 使用 Vortice 支持 Direct2D1 离屏渲染的更多相关文章

  1. iOS 离屏渲染的研究

    GPU渲染机制: CPU 计算好显示内容提交到 GPU,GPU 渲染完成后将渲染结果放入帧缓冲区,随后视频控制器会按照 VSync 信号逐行读取帧缓冲区的数据,经过可能的数模转换传递给显示器显示. G ...

  2. iOS离屏渲染简书

    更详细地址https://zsisme.gitbooks.io/ios-/content/chapter15/offscreen-rendering.html(包含了核心动画) GPU渲染机制: CP ...

  3. 离屏渲染学习笔记 /iOS圆角性能问题

    离屏渲染学习笔记 一.概念理解 OpenGL中,GPU屏幕渲染有以下两种方式: On-Screen Rendering 意为当前屏幕渲染,指的是GPU的渲染操作是在当前用于显示的屏幕缓冲区中进行. O ...

  4. 腾讯优测优分享 | 探索react native首屏渲染最佳实践

    腾讯优测是专业的移动云测试平台,旗下的优分享不定时提供大量移动研发及测试相关的干货~ 此文主要与以下内容相关,希望对大家有帮助. react native给了我们使用javascript开发原生app ...

  5. [ios]离屏渲染优化

    原文链接:https://mp.weixin.qq.com/s?__biz=MjM5NTIyNTUyMQ==&mid=2709544818&idx=1&sn=62d0d2e9a ...

  6. 探索react native首屏渲染最佳实践

    文 / 腾讯 龚麒 0.前言 react native给了我们使用javascript开发原生app的能力,在使用react native完成兴趣部落安卓端发现tab改造后,我们开始对由react n ...

  7. iOS性能优化中的离屏渲染

    GPU屏幕渲染有以下两种方式: On-Screen Rendering意为当前屏幕渲染,指的是GPU的渲染操作是在当前用于显示的屏幕缓冲区中进行. Off-Screen Rendering意为离屏渲染 ...

  8. OpenGL于MFC使用汇总(三)——离屏渲染

    有时直接创建OpenGL形式不适合,或者干脆不同意然后创建一个表单,正如我现在这个项目,创建窗体不显示,它仅限于主框架.而我只是ActiveX里做一些相关工作,那仅仅能用到OpenGL的离屏渲染技术了 ...

  9. opengl离屏渲染(不需要和窗口绑定,仅当作一个可以渲染一张图片的API使用)+ opencv显示

    具体过程参考的是这篇BLOG: http://wiki.woodpecker.org.cn/moin/lilin/swig-glBmpContext 这一片BLOG的代码有个 BOOL SaveBmp ...

  10. IOS 中openGL使用教程4(openGL ES 入门篇 | 离屏渲染)

    通常情况下,我们使用openGL将渲染好的图片绘制到屏幕上,但有时候我们不想显示处理结果,这时候就需要使用离屏渲染了. 正常情况下,我们将屏幕,也就是一个CAEAGLLayer对象作为渲染目标,离屏渲 ...

随机推荐

  1. 记录--怎么实现一个3d翻书效果

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 本篇主要讨论以下两种翻书动画的实现: 第一种是整页翻转的效果: 这种整页翻转的效果主要是做rotateY的动画,并结合一些CSS的3d属性 ...

  2. 记录--Event Loop事件循环、微任务、宏任务

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 前言 JS是一门单线程语言,单线程就意味着,所有的任务需要排队,前一个任务结束,才会执行下一个任务.这样所导致的问题是:如果JS执行的时间 ...

  3. 记录--三分钟打造自己专属的uni-app工具箱

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 介绍 可曾想过我们每次创建新项目,或者换地方写程序,都要把之前写过的工具类找出来又要复制粘贴一遍有些麻烦,尤其是写uni-app自定义模板 ...

  4. FFmpeg开发笔记(七)欧拉系统编译安装FFmpeg

    FFmpeg支持Linux.macOS.Windows.Android等操作系统,其中Linux系列包括Ubuntu.Debian.Mint.CentOS.RHEL.Fedora等分支.FFmpeg官 ...

  5. 马文·柯林斯的教育之道 「Marva Collins' Way」 阅读笔记

    <马文·柯林斯的教育之道> 是 哈佛幸福课(积极心理学) 中强列推荐的一本书 这本书主写的是 马文·柯林斯 的成长经历已经和教育理念,仅看介绍,会误认为这本书只是在说鼓励式的教学理念,看完 ...

  6. TorchV的RAG实践分享(三):解析llama_index的数据存储结构和召回策略过程

    1.前言 LlamaIndex是一个基于LLM的数据处理框架,在RAG领域非常流行,简单的几行代码就能实现本地的文件的对话功能,对开发者提供了极致的封装,开箱即用. 本文以官方提供的最简单的代理示例为 ...

  7. 用于vivado的TCL脚本创建工程

    用于实现在脚本所在的目录创建工程并初始化工程 点击查看代码 #获取脚本所在的文件 variable My_File [file normalize [info script]] #prj_path_f ...

  8. 基于rk3588----i2c驱动框架学习(2)-总线驱动 algorithm 分析

    rk3588 i2c algorithm 分析 来了来了,上次分析完i2c的驱动框架 今天我们就看看i2c的algorithm是如何实现的 static const struct i2c_algori ...

  9. 成长计划校园极客秀|基于OpenHarmony的智能阳台

    前言 本文由OpenAtom OpenHarmony(以下简称"OpenHarmony")开源开发者成长计划活动的参与者李建涛提供,详细阐述了由搭载OpenHarmony系统的拓维 ...

  10. Pandas通用函数和运算

    Pandas继承了Numpy的运算功能,可以快速对每个元素进行运算,即包括基本运算(加减乘除等),也包括复杂运算(三角函数.指数函数和对数函数等). 通用函数使用 apply和applymap app ...