『DotNetBrowser』.Net的浏览器嵌入组件,该选择DotNetBrowser 还是 CefSharp?

请添加图片描述
请添加图片描述
📣读完这篇文章里你能收获到

  • 全方位对比DotNetBrowserCefSharp的优缺点

请添加图片描述

文章目录

  • 一、引言
  • 二、引擎
  • 三、架构
    • 1. CefSharp架构
    • 2. DotNetBrowser架构
  • 四、对比
    • 1. 稳定性和内存使用
    • 2. 应用程序域
    • 3. AnyCPU
    • 4. H.264, AAC
    • 5. 安全
    • 6. Visual Studio设计器
    • 7. 嵌入应用程序 UI
    • 8. 高DPI
    • 9. 异步
    • 10. API和功能
      • 10.1 DOM访问
      • 10.2 与JavaScript交互
      • 10.3 从JavaScript调用.NET
      • 10.4 截屏
    • 11. 分发和部署
    • 12. 支持和更新
  • 五、总结

请添加图片描述

一、引言

将浏览器嵌入 .NET 应用程序中:DotNetBrowser 还是 CefSharp
image.png
为 WPF 或 WinForms 应用程序选择浏览器组件,对于那些搜索基于Chrome的解决方案的人来说,, DotNetBrowser和CefSharp是最明显的选择。
本文是在考虑其项目的开源库和商业库时提出的最常见比较点的汇总。

请添加图片描述

二、引擎

  • CefSharp 实际上是Chromium Embedded
    Framework (CEF) 的 .NET 包装器。包装通过C++/CLI 完成。
  • DotNetBrowser 在底层不使用 CEF 或 C++/CLI。相反,它采用了自己的方法直接与 Chromium 集成。它启动一个功能齐全的 Chromium 引擎,并通过进程间通信 (IPC) 与其进行通信。

请添加图片描述

三、架构

1. CefSharp架构

CefSharp 中,Chromium 引擎直接在.NET进程中初始化。初始化和关闭都必须在主应用程序线程(通常是 UI 线程)中执行。在不同的线程中调用它们通常会导致冻结。
此外,每个进程可以执行一次初始化和关闭。这个限制来自 CEF 本身。在执行关闭后尝试重新初始化 CefSharp 将导致错误。以下为CefSharp的架构图:
image.png

2. DotNetBrowser架构

DotNetBrowser 中,Chromium 引擎在单独的本机进程中进行初始化。不需要在主 UI 线程上执行此操作——即使在工作线程中也可以执行此操作。
可以同时初始化和使用具有不同配置的多个 Chromium 引擎,这在 CefSharp 中是不可能的。可以在不再需要 Chromium 时将其关闭并随时重新初始化。
DotNetBrowser架构图如下:
image.png

请添加图片描述

四、对比

1. 稳定性和内存使用

在单独的进程中运行 Chromium 有更多优点:

  1. 在这种情况下,内存消耗要低得多,这对于 32 位应用程序来说十分关键。
  2. CefSharp 中,如果 CEF 或 C++/CLI 绑定内部出现问题,这将导致整个 .NET 应用程序崩溃而无法处理这种情况。因为 .NET 应用程序可能会丢失或损坏用户的数据。
  3. 对于 DotNetBrowser,Chromium 内部的错误不会导致 .NET 应用程序崩溃。此外,甚至可以在托管代码中正确检测和处理这一切。例如,如果发生这种情况,那么可以重新初始化 Chromium 并恢复用户会话。

2. 应用程序域

由于架构原因,CefSharp不能在非默认AppDomain中使用。

  • 因此,CefSharp 不能用于通过 VSTO 插件或 Excel-DNA 将 Chromium 嵌入到 Office 应用程序中。 Office VSTO 将加载项加载到单独的AppDomain中以进行隔离。
  • DotNetBrowser 在非默认 AppDomain 中运行。可以在不同的 AppDomain 中创建多个 Chromium 引擎并同时使用它们。因此,DotNetBrowser 可用于创建 VSTO 加载项。

3. AnyCPU

在针对 AnyCPU 的应用程序中

  • 使用 CefSharp 时,你会发现它在这些应用程序的 64 位环境中无法正常工作。这儿有几个选项可以解决这个问题。其中之一是让你的应用程序始终在 32 位模式下运行,另一个更复杂,需要修改项目文件(.csproj或.vbproj)和代码。
  • DotNetBrowser 中,AnyCPU 支持开箱即用。因此,不需要类似的调整。

4. H.264, AAC

视频和音频通常使用专有编解码器进行编码,例如 H.264 和 AAC。

  • 此媒体无法在 CefSharp 中播放。要在 CefSharp 中启用这些编解码器,需要在启用专有编解码器的情况下自行重建 CEF。这是一项相当复杂的任务,可能需要长达一个月的时间。
  • DotNetBrowser 中默认禁用专有编解码器。可以通过编程方式启用它们,而无需重建库。
IEngine engine = EngineFactory.Create(new EngineOptions.Builder
{ProprietaryFeatures = ProprietaryFeatures.H264 | ProprietaryFeatures.Aac
}.Build());

5. 安全

Chromium 通过利用操作系统为它们提供的安全性来限制其渲染器和实用程序进程。此功能称为Chromium沙箱。其主要目的是防止第三方代码对计算机进行持久更改或访问机密信息。

  • CefSharp 不支持Chromium沙箱。这个限制来自 CEF 本身。
  • DotNetBrowser 支持沙箱并默认启用。如有必要,可以在初始化期间将其禁用。

而关于Chromium 中的漏洞

  • CefSharp 在 .NET 进程中启动 Chromium。这使你的应用程序容易受到 CEF 和 Chromium 中的漏洞的影响。如果恶意软件获得了对 Chromium 内存的访问权,它也会获得对 .NET 内存的访问权。
  • DotNetBrowser 在单独的进程中启动 Chromium。 Chromium 漏洞保留在 Chromium 中。

6. Visual Studio设计器

现代 WPF 和 Windows 窗体应用程序通常是在设计器的帮助下在 Visual Studio 中创建的。这种方法总体上简化了 UI 创建并节省了大量时间和精力。

  • CefSharp 提供有限的设计器支持。如果应用程序本身以 x86 为目标,则其控件将在设计器中正确处理。 AnyCPU 可能会工作,但尚未经过彻底测试。
  • DotNetBrowser 控件是纯 UI 控件,它们在代码中显式初始化。可以在设计器中不受任何限制地使用它们。安装 NuGet 包或 VSIX 扩展后,BrowserView 控件出现在工具箱中。它可以像任何其他常规 UI 控件一样被拖到窗体或窗口上。

7. 嵌入应用程序 UI

  • CefSharp 提供 WPF 和 Windows 窗体支持。但是,它的 WPF 实现只能在离屏渲染模式下工作。此实现具有有限的触摸屏和IME支持。
  • DotNetBrowser 在两种渲染模式下同时支持 WPF 和 Windows 窗体。在硬件加速模式下,触摸、手势和 IME 由 Chromium 自行处理,因此它们开箱即用。在离屏模式下,存在一些已知的限制。

以下是将 CefSharp 嵌入 WPF 窗口的方法:

<Window x:Class="CefSharpWpf.MainWindow"xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:wpf="clr-namespace:CefSharp.Wpf;assembly=CefSharp.Wpf"Title="MainWindow" Height="450" Width="800"><Grid><wpf:ChromiumWebBrowser Address="https://www.google.com"/></Grid>
</Window>
  • 就是这样,在最简单的情况下,不再需要编写代码。但是,在这种情况下,CefSharp 初始化和关闭是隐式执行的,很难确定它是否已经在某个点初始化。

DotNetBrowser 嵌入 WPF 窗口的过程需要额外的步骤。例如:
MainWindow.xaml:

<Windowxmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"xmlns:WPF="clr-namespace:DotNetBrowser.Wpf;assembly=DotNetBrowser.Wpf"x:Class="Embedding.Wpf.MainWindow"Title="MainWindow" Height="480" Width="800" Closed="Window_Closed"><Grid><WPF:BrowserView Name="browserView" /></Grid>
</Window>

MainWindow.xaml.cs:

public partial class MainWindow : Window
{private const string Url = "https://www.google.com";private readonly IBrowser browser;private readonly IEngine engine;public MainWindow(){// Create and initialize the IEngine instance.EngineOptions engineOptions = new EngineOptions.Builder{RenderingMode = RenderingMode.HardwareAccelerated}.Build();engine = EngineFactory.Create(engineOptions);// Create the IBrowser instance.browser = engine.CreateBrowser();InitializeComponent();// Initialize the WPF BrowserView control.browserView.InitializeFrom(browser);browser.Navigation.LoadUrl(Url);}private void Window_Closed(object sender, EventArgs e){browser?.Dispose();engine?.Dispose();}
}

在这里,大部分代码都与 Chromium 实例和IBrowser 实例的显式初始化和关闭有关。 UI 控件初始化是通过调用InitializeFrom()显式执行的。这种方法可以更好地控制初始化和关闭过程,并且更容易自定义初始 Chromium 配置。

8. 高DPI

  • CefSharp 中,浏览器子进程的默认 DPI 感知是 Per-Monitor。因此,桌面应用程序应具备DPI感知功能,才能在高 DPI 显示器(DPI 比例设置大于 100% 的显示器)上正确运行。在其他情况下,浏览器内容可能无法正确呈现,例如:

image.png

  • DotNetBrowser 以不同的方式支持高 DPI。在初始化过程中,它会检查当前进程的 DPI 感知,并为相应的 Chromium 引擎设置匹配的 DPI 感知。因此,无需让应用程序显式识别 DPI 以避免在高 DPI 显示上呈现伪影。

image.png

9. 异步

DotNetBrowser 和 CefSharp 都可以在没有 UI 的应用程序中使用。

  • CefSharp 中,CefSharp.OffScreen.ChromiumWebBrowser用于此目的。初始化过程通常保持不变。但是,如果代码使用 async/await 模式,则需要使用同步上下文来确保在主线程上而不是在不同的工作线程上执行初始化和关闭。
  • 要在没有 UI 的应用程序中使用 DotNetBrowser,需要像往常一样执行初始化。在这种情况下,没有需要初始化的BrowserView。即使代码使用async/await模式,也无需创建和使用同步上下文。

10. API和功能

这两种产品都有许多可用的功能。在本文中,我将比较几个最重要的,以展示 API 的不同之处。

10.1 DOM访问

  • CefSharp 中,只能通过执行 JavaScript 调用来访问 DOM。
var script = @"document.getElementsByName('question')[0].value = 'CefSharp Example';document.getElementsByName('btn')[0].click();
";
browser.ExecuteScriptAsync(script);
  • DotNetBrowser 提供了丰富的 DOM API,可用于直接从 .NET 执行以下操作:
IDocument document = browser.MainFrame.Document;
(document.GetElementByName("question") as IInputElement).Value = "DotNetBrowser Example";
document.GetElementByName("btn").Click();
  • 访问和修改 DOM 树
  • 更改 HTML 元素属性
  • 订阅 DOM 事件并从 .NET 代码中调度它们

因此,在 DotNetBrowser 中与网页执行复杂的交互要方便得多。无需编写难以调试和支持的复杂 JavaScript 代码。 DotNetBrowser 中的 DOM API 不是一组 JavaScript 调用的包装器。它直接对 Blink 引擎进行 IPC 调用。

10.2 与JavaScript交互

执行JavaScript并处理结果

  • CefSharp 和 DotNetBrowser 都提供了在网页上执行 JavaScript 的能力。在 CefSharp 中,有两种方法可用于此目的,ExecuteJavaScriptAsync和EvaluateScriptAsync。两者都可用于浏览器本身(通过扩展方法)或其中的一个框架:
// Execute JavaScript without returning a result. The method returns 
// before the script has actually been executed.
browser.ExecuteJavaScriptAsync("alert('All Resources Have Loaded');");// Evaluate some Javascript code. The script will be executed asynchronously 
// and the method returns a Task encapsulating the response from the 
// JavaScript.
JavascriptResponse response = await browser.EvaluateScriptAsync(script);

然后使用JavascriptResponse.Result获取执行结果。可能的结果类型有 bool, int, long, double, string, List, IDictionary<string, object>, 和IJavascriptCallback。这里的集合是 JavaScript 集合的快照表示,而IJavascriptCallback是一种 JavaScript 函数表示,可用于从 .NET 端执行它。

  • DotNetBrowser 中,有 IFrame.ExecuteJavaScript()用于此目的。此方法的通用版本可用于显式指定预期的返回类型:
string title = await browser.MainFrame.ExecuteJavaScript<string>("document.title");
IJsObject window = await browser.MainFrame.ExecuteJavaScript<IJsObject>("window");
IElement body = await browser.MainFrame.ExecuteJavaScript<IElement>("document.body");

这里的主要区别是可以将 JavaScript 对象表示为IJsObject。使用此接口,可以访问和修改 JavaScript 对象的属性并调用其方法。在 .NET 端对IJsObject所做的所有更改都将立即反映在 JavaScript 端。此外,ExecuteJavaScript调用可以返回一个IElement,这是一个 DOM 元素的表示,可以使用它来访问和修改 DOM 属性或订阅 DOM 事件。

10.3 从JavaScript调用.NET

CefSharp 和 DotNetBrowser 都可以使网页上的 JavaScript 可以访问 .NET 对象,但是,CefSharp 存在一些特定的限制。

  • CefSharp JavaScript 绑定可用于 JavaScript 和 .NET 之间的通信。但是,CefSharp 不允许将Form, Window或任何Control注入 JavaScript。另外,CefSharp 只支持调用注入对象的方法。如果需要设置属性,则必须修改类并创建 Get/Set 方法。
  • DotNetBrowser 中,可以将任何对象注入 JavaScript,包括 Form, Window和Control对象。执行注入后,可以访问注入的 .NET 对象的公共字段、属性和方法。此外,DotNetBrowser 支持从 JavaScript 访问索引属性(使用字符串或数字索引器)。如果需要从 JavaScript 访问 .NET 集合,这会很有帮助。

10.4 截屏

两种解决方案都支持在浏览器不可见时进行截屏。但是,API 有明显不同。以下是代码片段:
在CefSharp中截图:

// Take a screenshot
var bitmapAsByteArray = await browser.CaptureScreenshotAsync();// Save the screenshot as PNG
var screenshotPath = Path.GetFullPath("screenshot.png");
File.WriteAllBytes(screenshotPath, bitmapAsByteArray);

在DotNetBrowser中截图:

// Take a screenshot
DotNetBrowser.Ui.Bitmap image = browser.TakeImage();// Convert the screenshot to System.Drawing.Bitmap and save it as PNG
System.Drawing.Bitmap bitmap = image.ToBitmap();
bitmap.Save("screenshot.png", ImageFormat.Png);

主要的 DotNetBrowser DLL 不使用System.Drawing中的类型,因为它的限制,因此,它提供了自己的类型。然后可以通过DotNetBrowser.Wpf或DotNetBrowser.WinForms中提供的扩展方法将此类型转换为常规System.Drawing.Bitmap。

11. 分发和部署

  • CefSharp 需要Microsoft Visual C++运行时存在于环境中。 Visual C++ 2015 是最低版本,但所需的确切版本取决于 Chromium 版本。因此,需要在希望运行基于 CefSharp 的应用程序的每台机器上预安装 Microsoft Visual C++ Redistributable Package,将其设置为安装程序的依赖项,或将其 DLL 打包为应用程序的一部分,并确保 CefSharp 正确找到它们。
  • DotNetBrowser 中,所有必需的 Chromium 二进制文件和 DLL 都已打包到 DotNetBrowser DLL 中,并且可以在执行期间自动提取。无需预先安装 Microsoft Visual C++ Runtime 即可使用 DotNetBrowser。

12. 支持和更新

  • CefSharp 是一个开源项目。如果发现错误或缺少功能,可以提出建议。
  • DotNetBrowser是为使用.NET开发软件的商业公司设计和创建的商业产品,对集成第三方解决方案的质量和支持有很高的要求。自 2015 年以来,TeamDev 开发并支持 DotNetBrowser。

所有已订阅有效标准支持的客户都可免费使用所有DotNetBrowser 新版本并获得技术支持。如果发现错误或缺少功能,提交反馈即可。
几乎每个月都会发布一个新版本的 DotNetBrowser。会在 Chromium 正式发布后的 3-4 周内将 Chromium 升级到最新的稳定版本(带有最新的安全补丁和修复的漏洞)。

请添加图片描述

五、总结

因为开源和免费,CefSharp 被广泛使用。它很容易为基本案例进行配置,并且拥有广泛的文档和活跃的开源开发者社区。
但是,它具有来自其设计和架构的限制。缺少沙盒支持使其安全性降低,并且进程内方法会影响稳定性并增加应用程序的内存使用量。
它也不能用于暗示在单独的 AppDomain(如 VSTO)中运行代码的环境。对于其他一些情况,例如播放使用专有编解码器编码的内容,必须自己构建、更新和维护 CEF。这需要大量额外的工作和基础设施。
与加载的网页的复杂交互会通过 JavaScript 注入执行,这使得生成的代码更难调试和支持。
DotNetBrowser 简化了所有这些案例的开发过程——由于它的进程外架构,它可以用于创建更稳定和安全的解决方案。使用 DotNetBrowser,可以与需要在单独的 AppDomain 中运行代码的应用程序集成,并在网页上执行复杂的操作,而无需进行大量的 JavaScript 注入。
除此之外,如果你觉得缺少某样功能、发现某些问题或有其他疑问,你可以随时与DotNetBrowser团队联系并获得帮助。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/14305.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Ubuntu 22.04.2 LTS 安装搜狗输入法后,修改区域格式Regional Format crash 崩溃 ,改用bash 指令修改

故障 系统已经升级到最新 基于Ubuntu 20.04 LTS apt upgrade升级而来。 yeqiangyeqiang-MS-7B23:~$ cat /etc/os-release PRETTY_NAME"Ubuntu 22.04.2 LTS" NAME"Ubuntu" VERSION_ID"22.04" VERSION"22.04.2 LTS (Jammy Jellyfish)"…

JavaWeb 笔记——5

JavaWeb 笔记——5 一、Filter1.1、概述1.2、Filter快速入门1.3、Filter执行流程1.4、Filter使用细节1.5、Filter-案例-登陆验证 二、Listener2.1、Listener概述与分类2.2、ServletContextListener使用 三、AJAX3.1、AJAX概述3.2、AJAX快速入门3.3、使用Ajax验证用户名是否存在…

DAY36:贪心算法(三)最大子数组和+买卖股票最佳时机

文章目录 53.最大子数组和枚举思路暴力解法贪心思路完整版时间复杂度 122.买卖股票的最佳时机Ⅱ&#xff08;解法比较巧妙&#xff09;思路完整版总结 53.最大子数组和 给你一个整数数组 nums &#xff0c;请你找出一个具有最大和的连续子数组&#xff08;子数组最少包含一个元…

论文解读:PeSTo:用于精确预测蛋白质结合界面的无参数几何深度学习

Title:PeSTo: parameter-free geometric deep learning for accurate prediction of protein binding interfaces 期刊&#xff1a;nature communication 分区&#xff1a;一区 影响因子&#xff1a;16.6 webserver:t Pesto Github:GitHub - LBM-EPFL/PeSTo 摘要 蛋白质是…

Spring Boot 中的 CompletableFuture 类是什么,如何使用?

Spring Boot 中的 CompletableFuture 类是什么&#xff0c;如何使用&#xff1f; 介绍 在开发企业级应用程序时&#xff0c;我们经常需要异步执行任务。异步执行任务可以提高应用程序的性能和响应能力。在 Java 8 中&#xff0c;引入了 CompletableFuture 类&#xff0c;它提…

git no matching host key type found. Their offer: ssh-rsa

本地生成ssh&#xff0c;并配置了服务器SSH Public Keys 问题 拉去远程代码报错 git no matching host key type found. Their offer: ssh-rsa (base) tangsiqitangsiqideMacBook-Pro VDI % git clone "ssh://tangsiqireview.archeros.cn:29418/ArcherDT/vdi-server&…

Centos 7 下安装Redis

官网地址&#xff08;英文&#xff09;&#xff1a;Redis 官网地址&#xff08;中文&#xff09;&#xff1a;CRUG网站 or redis中文文档 Redis源码地址&#xff1a;GitHub - redis/redis: Redis is an in-memory database that persists on disk. The data model is key-v…

500万PV的网站需要多少台服务器?

1. 衡量业务量的指标 衡量业务量的指标项有很多&#xff0c;比如&#xff0c;常见Web类应用中的PV、UV、IP。而比较贴近业务的指标项就是大家通常所说的业务用户数。但这个用户数比较笼统&#xff0c;其实和真实访问量有比较大的差距&#xff0c;所以为了更贴近实际业务量及压力…

非主流币波段策略

数量技术宅团队在CSDN学院推出了量化投资系列课程 欢迎有兴趣系统学习量化投资的同学&#xff0c;点击下方链接报名&#xff1a; 量化投资速成营&#xff08;入门课程&#xff09; Python股票量化投资 Python期货量化投资 Python数字货币量化投资 C语言CTP期货交易系统开…

五.LLC谐振变换器

LLC 谐振变换器启动过程分析 LLC 谐振变换器的组成结构中包含容性器件&#xff0c;为了尽可能减小输出电压纹波&#xff0c;钳位输出电压&#xff0c;此时希望输出滤波电容尽可能的大&#xff0c;因此也会在启动的时候&#xff0c;电容两端电压近似为 0&#xff0c;系统对电容…

Vue3.3 编译宏

Vue 3.3新增了一些语法糖和宏&#xff0c;包括泛型组件、defineSlots、defineEmits、defineOptions defineProps 父子组件传参 <template><div><Child name"xiaoman"></Child></div> </template><script langts setup>…

装饰器模式:通过剖析Java IO类库源码学习装饰器模式

我们通过剖析Java IO类的设计思想&#xff0c;再学习一种新的结构型模式&#xff0c;装饰器模式。它的代码结构跟桥接模式非常相似&#xff0c;不过&#xff0c;要解决的问题却大不相同。 Java IO类库非常庞大和复杂&#xff0c;有几十个类&#xff0c;负责IO数据的读取…