性能测试监控指标及分析调优 | 京东云技术团队一、哪些因素会成为系统的瓶颈?

1. 什么是MAF和MEF?

MEF和MEF微软官方介绍:Managed Extensibility Framework (MEF) - .NET Framework | Microsoft Learn

MEF是轻量化的插件框架,MAF是复杂的插件框架。

因为MAF有进程隔离和程序域隔离可选。我需要插件进程隔离同时快速传递数据,最后选择了MAF。

如果不需要真正的物理隔离还是建议使用简单一点的MEF框架。

2.什么情况下会使用MAF框架?

例如想做一个浏览器,每点开一个页面都是一个独立的进程,然后进程之间相互隔离。或者多个不同的进程到同一个界面渲染,实现后台进程和界面之间的隔离。

多进程之间相互传递数据也可以使用MAF。

3. 如何学习MAF?

MAF其实是一项很老的技术,入门我看的是《WPF编程宝典》第32章 插件模型。里面有MAF和MEF的详细介绍和许多样例。

但是要深入理解还是看了很多其他的东西,下面我详细说明,我自己理解和总结的MAF。

4. MAF框架入门

4.1 MAF框架构成与搭建

MAF框架模式是固定的,这里做一个详细介绍。

首先是要添加几个新项目,下图中不包含主项目。

Addin文件夹是放置插件用的,其余都是必要项目。

假设HostView项目和主项目的输出路径是..\Output\

然后修改每个项目的输出文件夹,例如AddInSideAdapter项目输出路径可以设置为..\Output\AddInSideAdapters\

注意插件项目输出到Addin文件夹中的子文件夹是..\..\Output\AddIns\MyAddin\

最后项目的输出文件夹结构是:

D:\Demo\Output\AddIns

D:\Demo\Output\AddInSideAdapters

D:\Demo\Output\AddInViews

D:\Demo\Output\Contracts

D:\Demo\Output\HostSideAdapters

来看看MAF框架模型构成。

 上图中绿色的是被引用蓝色项目所引用。例如HostSideAdapter就要引用Contract和Hostview,如下图所示。

 注意引用时取消勾选复制本地。

 这时就完成基本项目结构的搭建。

4.2 MAF框架实现

这里想实现宿主项目和插件项目的双向通信。即插件项目将相关函数接口在宿主实现,然后将宿主项目相关函数接口用委托类的方式注册给插件项目。实现双向通信。

用《WPF编程宝典》样例代码来说,样例中,插件程序实现ProcessImageBytes处理图像数据的函数,处理同时需要向宿主项目报告处理进度,宿主中 ReportProgress函数实现进度可视化。

MAF实现一般是先写Contract协议,明确需要的函数接口。然后写AddlnView和HostView。实际上这两个是将函数接口抽象化,在接口里函数复制过来前面加 public abstract 就行。

之后HostSideAdapter和AddInSideAdapter直接快速实现接口。

首先从Contract开始,Contract是定义接口,需要设置对象标识符[AddInContract],且必须继承IContract。

 [AddInContract]public interface IImageProcessorContract : IContract{byte[] ProcessImageBytes(byte[] pixels);void Initialize(IHostObjectContract hostObj);}public interface IHostObjectContract : IContract{void ReportProgress(int progressPercent);}

Initialize函数是提供宿主函数注册的接口。

 然后在HostView和AddInView分别定义主程序和插件程序的接口抽象类。

public abstract class ImageProcessorHostView{public abstract byte[] ProcessImageBytes(byte[] pixels);public abstract void Initialize(HostObject host);}public abstract class HostObject{public abstract void ReportProgress(int progressPercent);}

注意AddlnView需要设置对象标识符[AddInBase]。

 [AddInBase]public abstract class ImageProcessorAddInView{public abstract byte[] ProcessImageBytes(byte[] pixels);public abstract void Initialize(HostObject hostObj);}public abstract class HostObject{public abstract void ReportProgress(int progressPercent);}

之后在HostSideAdapter实现抽象类。

注意HostSideAdapter继承HostView的抽象类,在构造函数里需设置ContractHandle插件生存周期,ContractHandle不能为readonly。

  [HostAdapter]public class ImageProcessorContractToViewHostAdapter : HostView.ImageProcessorHostView{private Contract.IImageProcessorContract contract;private ContractHandle contractHandle;public ImageProcessorContractToViewHostAdapter(Contract.IImageProcessorContract contract){            this.contract = contract;contractHandle = new ContractHandle(contract);}              public override byte[] ProcessImageBytes(byte[] pixels){return contract.ProcessImageBytes(pixels);}public override void Initialize(HostView.HostObject host){            HostObjectViewToContractHostAdapter hostAdapter = new HostObjectViewToContractHostAdapter(host);contract.Initialize(hostAdapter);}}public class HostObjectViewToContractHostAdapter : ContractBase, Contract.IHostObjectContract{private HostView.HostObject view;public HostObjectViewToContractHostAdapter(HostView.HostObject view){this.view = view;}public void ReportProgress(int progressPercent){view.ReportProgress(progressPercent);}        }

在AddInSideAdapter实现Contract接口,基本和HostSideAdapter类似,只是继承的类不同。

[AddInAdapter]public class ImageProcessorViewToContractAdapter : ContractBase, Contract.IImageProcessorContract{private AddInView.ImageProcessorAddInView view;public ImageProcessorViewToContractAdapter(AddInView.ImageProcessorAddInView view){this.view = view;}public byte[] ProcessImageBytes(byte[] pixels){return view.ProcessImageBytes(pixels);}public void Initialize(Contract.IHostObjectContract hostObj){            view.Initialize(new HostObjectContractToViewAddInAdapter(hostObj));            }}public class HostObjectContractToViewAddInAdapter : AddInView.HostObject{private Contract.IHostObjectContract contract;private ContractHandle handle;public HostObjectContractToViewAddInAdapter(Contract.IHostObjectContract contract){this.contract = contract;this.handle = new ContractHandle(contract);            }public override void ReportProgress(int progressPercent){contract.ReportProgress(progressPercent);}}

宿主项目中需要实现HostView里HostObject抽象类。

 private class AutomationHost : HostView.HostObject{private ProgressBar progressBar;public AutomationHost(ProgressBar progressBar){this.progressBar = progressBar;}public override void ReportProgress(int progressPercent){// Update the UI on the UI thread.progressBar.Dispatcher.BeginInvoke(DispatcherPriority.Normal,(ThreadStart)delegate(){progressBar.Value = progressPercent;});                }}

然后是在宿主项目里激活插件,并初始化AutomationHost。

string path = Environment.CurrentDirectory;            
AddInStore.Update(path);//更新目录中Addins目录里的插件
IList<AddInToken> tokens = AddInStore.FindAddIns(typeof(HostView.ImageProcessorHostView), path);//查找全部插件
lstAddIns.ItemsSource = tokens;//插件可视化
AddInToken token = (AddInToken)lstAddIns.SelectedItem;//选择插件
AddInProcess addInProcess = new AddInProcess();//创建插件进程
addInProcess.Start();//激活插件进程
addin = token.Activate<HostView.ImageProcessorHostView>(addInProcess,AddInSecurityLevel.Internet);//激活插件
//如果只是想隔离程序域,就无需创建AddInProcess,激活插件如下
// HostView.ImageProcessorHostView addin = token.Activate<HostView.ImageProcessorHostView>(AddInSecurityLevel.Host);
automationHost = new AutomationHost(progressBar);//创建AutomationHost类
addin.Initialize(automationHost);//初始化automationHost

 插件项目中实现AddInView中的抽象类。

[AddIn("Negative Image Processor", Version = "1.0", Publisher = "Imaginomics",Description = "")]public class NegativeImageProcessor : AddInView.ImageProcessorAddInView {public override byte[] ProcessImageBytes(byte[] pixels){int iteration = pixels.Length / 100;         for (int i = 0; i < pixels.Length - 2; i++){pixels[i] = (byte)(255 - pixels[i]);pixels[i + 1] = (byte)(255 - pixels[i + 1]);pixels[i + 2] = (byte)(255 - pixels[i + 2]);if (i % iteration == 0){host?.ReportProgress(i / iteration);}}return pixels;}private AddInView.HostObject host;public override void Initialize(AddInView.HostObject hostObj){host = hostObj;}

 这时宿主可以把数据传递给插件程序,插件程序中ProcessImageBytes处理数据然后通过host?.ReportProgress(i / iteration);向宿主传递消息。

这里有提供样例程序。

项目附件.7z

5. MAF框架常见问题

5.1手动关闭插件

AddInController addInController = AddInController.GetAddInController(addIn);
addInController.Shutdown();

此方法适应于非应用隔离的手动关闭。对于应用隔离式插件,用此方法会抛出异常。

如上面样例就是应用隔离的插件,可以根据进程id直接关闭进程。

public void ProcessClose(){try{if (process != null){Process processes = Process.GetProcessById(addInProcess.ProcessId);if (processes?.Id > 0){processes.Close();}}}catch (Exception){}5.2 插件异常

System.Runtime.Remoting.RemotingException: 从 IPC 端口读取时失败: 管道已结束。这是插件最常见的异常,因为插件抛出异常而使得插件程序关闭。

如果是插件调用非托管代码,而产生的异常,可以查Windows应用程序日志来确定异常。其余能捕获的异常尽量捕获保存到日志,方便查看。

5.3 双向通信

实际应用过程中,往往是通过委托来将宿主相关函数暴露給一个类,然后通过在宿主程序初始化后。在插件中实例化后就可以直接调用宿主的相关函数,反之同理。

这里是通过委托暴露宿主的一个函数。

public delegate void UpdateCallBack(string message, bool isclose, int leve);public class VideoHost : HostAddInView{public event UpdateCallBack Updatecallback;public override void ProcessVideoCallBack(string message, bool isclose, int leve){Updatecallback?.Invoke(message, isclose, leve);}}

 在插件程序中实例化后调用。

private HostAddInView hostAddInView;public override void Initialize(HostAddInView hostAddInView){this.hostAddInView = hostAddInView;}private void ErrorCallback(string message, bool isclose, int leve){hostAddInView?.ProcessVideoCallBack(message, isclose, leve);}

6. MAF深入理解

 MAF本质是实现IpcChannel通信,在一个期刊中有作者抛弃MAF固定结构自己实现IpcChannel,因为代码很复杂,就不在此详细阐述。

如果要实现应用域隔离,自己实现IpcChannel,MAF中的应用域隔离实现也是非常好的参考资料。

MAF的7层结构主要是实现从插件的宿主函数转换,例如可以在将插件程序的界面放入主界面中渲染,做出像浏览器一样的开一个界面就是一个进程。将插件中的组件在AddInSideAdapter中转换为Stream然后在HostSideAdapter中将Stream实例化为组件。而HostView和AddInView实际上是提供两个转换接口,Contract是定义传输接口。

另外如果传输插件向数组传递图像数据,最后是转换成byte[],或者使用共享内存。 

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

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

相关文章

基于Spring Boot的网上购物商城系统

目录 前言 一、技术栈 二、系统功能介绍 用户功能模块的实现 管理员功能模块的实现 商家功能模块的实现 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 本课题是根据用户的需要以及网络的优势建立的一个基于Spring Boot的网上购物商城系统&#xff0c;…

ubuntu、linux in window安装docker教程

1、首先进入管理员权限。 2、更新软件源。 sudo apt update 3、安装一些依赖 sudo apt install apt-transport-https ca-certificates curl software-properties-common 4、为系统添加Docker的密钥 curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-k…

【RabbitMQ实战】docker 安装RabbitMQ(bitnami)

一、搜索镜像 [rootlocalhost ~]# docker search rabbitmq NAME DESCRIPTION STARS OFFICIAL AUTOMATED rabbitmq RabbitMQ is an open source multi-pro…

【论文笔记】Scene as Occupancy

原文链接&#xff1a;https://arxiv.org/abs/2306.02851 1. 引言 与传统的3D框物体表达相比&#xff0c;使用3D占用表达是几何感知的&#xff0c;因为3D框表达简化了物体的形状。此外&#xff0c;现有基于视觉的方法很少考虑时间信息&#xff1b;单阶段方法缺少从粗到细的细化…

建议收藏《Verilog代码规范笔记_华为》(附下载)

华为verilog编程规范是坊间流传出来华为内部的资料&#xff0c;其贴合实际工作需要&#xff0c;是非常宝贵的资料&#xff0c;希望大家善存。至于其介绍&#xff0c;在此不再赘述&#xff0c;大家可看下图详细了解&#xff0c;感兴趣的可私信领取《Verilog代码规范笔记_华为》。…

open62541开发:添加sqlite3 历史数据库

历史数据库在OPCUA 应用中十分重要&#xff0c;例如OPCUA 网关和OPCUA 汇聚服务器中都需要历史数据库功能。但是open62541 协议栈中仅包含了基于内存的历史数据库&#xff0c;在实际应用中是不够的。本博文讨论open62541 中添加sqlite3 为基础的历史数据库若干问题。 借鉴 Gi…

Postman应用——接口请求和响应(Get和Post请求)

文章目录 新增Request请求Get请求Post请求 Request请求响应Postman响应界面说明请求响应另存为示例&#xff08;模板&#xff09;Postman显示的响应数据清空请求响应数据保存到本地文件 这里只讲用的比较多的Get和Post请求方式&#xff0c;也可以遵循restful api接口规范&#…

【Spring Boot】拦截器学习笔记

一、普通拦截器 1&#xff0c;新建类MyWebConfig实现WebMvcConfigurer&#xff0c;实现addInterceptors方法 Overridepublic void addInterceptors(InterceptorRegistry registry) {registry// 不拦截哪些请求.excludePathPatterns("/login")// 拦截哪些请求.addPat…

D1117内置热保护和电流限制保护功能,输出电流能力为 1.0A,应用于计算机主板和显卡电源管理等产品上

D1117 是一款低压差线性稳压电路&#xff0c;该电路输出电流能力为 1.0A。该系列电路包含固定输出电压版本和可调输出电压版本&#xff0c; 其输出电压精度为1.5% 。为了保证芯片和电源系统的稳定性&#xff0c;D1117 内置热保护和电流限制保护功能&#xff0c;同时产品采用了…

【前端知识】Three 学习日志(七)—— 动画渲染循环

Three 学习日志&#xff08;七&#xff09;—— 动画渲染循环 一、旋转动画 // 渲染函数 function render() {renderer.render(scene, camera); //执行渲染操作mesh.rotateY(0.01);//每次绕y轴旋转0.01弧度requestAnimationFrame(render);//请求再次执行渲染函数render&#…

Microsoft 网络监控

随着网络的发展和变得越来越复杂&#xff0c;公司比以往任何时候都更需要监控其网络基础设施&#xff0c;因为即使是轻微的系统中断也可能导致重大损失。网络监控工具提供实时数据和网络状态的图形概述。这使您能够准确地了解正在发生的事情&#xff0c;以便您知道需要更改的位…

基于Java的公务员考试资料共享平台的设计与实现

前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战✌&#x1f497; &#x1f447;&#x1f3fb;…

vue管理系统列表行按钮过多, 封装更多组件

管理系统table列表操作列, 随着按钮的数量越来越多会不断加宽操作列, 感觉很不好, 对此我封装了这个自动把多余的按钮放到更多菜单下 MoreOperation/index.vue menu组件我这是ant的, 可以自行替换为其他框架的 <template><div class"table-operations-group"…

CSS 实现祥云纹理背景

&#x1fab4; 背景 最近掘金出来一个中秋创意活动&#xff0c;我准备参加一下。作品方向选择用纯css做一个中秋贺卡&#xff0c;其中有一些中秋的元素和一些简单的动画&#xff0c;而贺卡背景的实现就是本文要讲的内容。 中秋贺卡成果图&#xff08;生成gif有点失真&#x1f6…

网站有反爬机制就爬不了数据?那是你不会【反】反爬

目录 前言 一、什么是代理IP 二、使用代理IP反反爬 1.获取代理IP 2.设置代理IP 3.验证代理IP 4.设置代理池 5.定时更新代理IP 三、反反爬案例 1.分析目标网站 2.爬取目标网站 四、总结 前言 爬虫技术的不断发展&#xff0c;使得许多网站都采取了反爬机制&#xff…

Python150题day08

2.基础语法篇 2.1 if 条件句 ①单个条件分支 使用input函数接收用户的输入&#xff0c;如果用户输入的整数是偶数&#xff0c;则使用print函数输出"你输入的整数是:{value],它是偶数”&#xff0c;[value]部分要替换成用户的输入。 解答: value input("请输⼊⼀…

ESP8266 WiFi物联网智能插座—项目简介

目录 1、项目背景 2、设备节点功能 3、上位机功能 物联网虽然能够使家居设备和系统实现自动化、智能化管理&#xff0c;但是依然需要依靠更为先进的终端插座作为根本保障&#xff0c;插座是所有家用电器需要使用的电源设备&#xff0c;插座的有序智能管理&#xff0c;对于实…

Ubuntu系统下载及安装教程

史上最全最新Ubuntu安装教程&#xff08;图文&#xff09; - 知乎 (说明&#xff1a;本教程介绍的是安装DeskTop版的系统) 1.官网下载镜像 官方网址: https://ubuntu.com/#download进入官网后会有最新版本的镜像下载地址&#xff0c;如果需要下载最新版本&#xff0c;直接点…

Openresty(二十二)ngx.balance和balance_by_lua终结篇

一 灰度发布铺垫 ① init_by_lua* init_by_lua init_by_lua_block 特点: 在openresty start、reload、restart时执行,属于master init 阶段机制&#xff1a; nginx master 主进程加载配置文件时&#xff0c;运行全局Lua VM级别上的参数指定的Lua代码场景&#xff1a; …

SadTalker 让图片说话

参考&#xff1a;https://github.com/OpenTalker/SadTalker 其他类似参考&#xff1a;https://www.d-id.com/ 输入图片加音频产生2d视频 安装使用 1、拉取github&#xff0c;下载对应安装库 2、下载对应模型baidu网盘 新建checkpoints&#xff0c;把下载sadtalker里模型拷贝进…