C#+WPF上位机开发(模块化+反应式)

在上位机开发领域中,C#与C++两种语言是应用最多的两种开发语言,在C++语言中,与之搭配的前端框架通常以QT最为常用,而C#语言中,与之搭配的前端框架是Winform和WPF两种框架。今天我们主要讨论一下C#和WPF这一对组合在上位机开发过程中的实际应用。

在这里插入图片描述

一、模块化概念

开发一套完善的软件,离不开良好的架构,而说到架构,在C#中,Prism框架与WPF的配合可谓大大提高了开发效率,原因如下:Prism框架中有IOC容器(分别是unity和dryioc),IOC容器负责管理类的生命周期;另外,Prism框架的WPF版本还有区域管理器(RegionManager)、事件聚合器(EventAggregator),对话框服务(DialogService)、适配器(Adapter)、模块管理器(ModuleManager)等。

除了模块管理器(IModule接口),其他几个重要的管理器都是为了更方便实现界面呈现及业务逻辑开发,那么模块管理器的功能是什么呢?程序员往往会将一个大型的软件项目拆分成小项目(模块),比如上位机系统中会接入各种硬件(工业相机、PLC、数据采集器、扫码枪、控制卡等),良好的软件架构思路是将同一种类型的硬件抽象成一个硬件抽象类,然后,不同型号的硬件与继承这个抽象基类,实现各自硬件的业务逻辑(加载硬件、打开硬件、使用硬件、关闭硬件),要实现这些开发,Prism的模块就派上用场了。也就是将硬件进行模块化,由Prism框架去实现及统一管理。

当然,除了上述的例子,软件的基础库,软件的业务逻辑,同样可以拆成不同的模块,最后由Prism框架统一管理。

二、反应式概念

C#语言为我们提供了一个IObservable接口,也就是设计模式中的观察者模式在C#语言中的具体实现。ReactiveUI框架对此接口进行了扩展,使之变得易用。

在传统的软件开发中,我们对某个属性发生变化后要进行下一步的相应处理时,会主动对判断这个属性的值,然后实现相应的代码逻辑。而在观察者模式下,我们会写一个观察者,由这个观察者去“实时”观察这个属性,一旦观察到属性值发生了改变,会抛出一个通知。

那么,通知谁呢?

谁去订阅了这个观察者,其回调函数都会被触发一次调用,从而达到一个目的,即被观察的那个属性发生改变后会执行事先写好的“某一段代码逻辑”。

这样的开发模式在写上位机软件时非常可靠和易用,因为上位机可能会实时监测下位机的某些参数变化,一旦硬件参数发生变化,软件要相应给出动作(做出反应),这种反应,我们称为反应式编程。

三、软件架构

良好的架构,可以帮助开发人员提高开发效率,减少bug的发生,增强系统的稳定性。我们可以将一个上位机系统分成如下几个方面:

第一是具有通用性的基础类库,这一些类与具体的业务逻辑无关,只帮助处理业务过程中的数据,通常是一些帮助类,或者语言包,本地设置等等;

第二、是硬件模块,上位机开发离不开硬件的接入,而不同的上位机系统,其接入的硬件也是五花入门,比如MES系统,可能最常见的硬件是PLC、扫码枪;而控制系统,则控制卡与各种被控设备的信息反馈最为常见;AOI,AXI等工业视觉检测方面,则接入的硬件除了PLC和控制卡,还有工业相机,光机,光源等等;或者医学方机的上位机,包含了更多的非标设备。

如此多的硬件,如何有序的接入到上位机呢?

答案是,最好将同一种类型的硬件抽象成一个基类,通过接口、抽象类、继承等方面的编程手段,为每一种不同厂家或不同型号的硬件创建一个项目,将这些项目看成是一个个独立的模块,像小孩子玩积木游戏一般,将硬件“堆积”到上位机中,一旦某个型号的硬件想被替换,直接将新硬件的模块加载到上位机即可。

第三,是数据库模块。由于上位机系统运行过程中需要处理大量的数据,其处理结果需要持久化,一般采用文件或数据库的形式进行保存。而数据库的操作就可以单独形成一个模块,这个模块包含抽象类模块和具体的数据库操作类模块,因为最终要保存的数据库可能是mysql、oracle、mssql等不同的数据库,所以最好的方式是为不同类型的数据库的增删改查等操作开发各自的模块,以便管理和替换。

第四,是业务逻辑模块。这一类型的模块会随着不同行业的上位机功能需求,模块的多少会有所不同。尽量将复杂的业务逻辑拆分成独立的小型业务模块,模块与模块之间采用接口通讯,这样做的好处是:减少代码间的耦合程度,尽量开发一些职责单一的类,达到具有良好扩展性的代码架构。
在这里插入图片描述
四、详细设计

在聊完了大致的软件架构,我们来谈谈具体的模块设计。利用vs2022开发软件,一个模块就是一个项目,有的项目是类库,有的是用户控件,有的是窗体,还有的是C++项目,具体的开发如下所示:

为了更清晰明了每个项目的含义,我们制定一套项目(模块)的命名规则:公司名+架构分类名+模块名,比如Company.Application.Main,表示应用层的主模块。

第一、程序入口项目

Company.Application.Shell,这是上位机系统的入口,其实是一个窗体,启动时加载的第一个窗体,其内容由主模块进行填充。

第二、应用层模块

Company.Application.Main,这是上位机的主模块,由Prism框架的区域管理器将此模块导航到Shell窗体中。

Company.Application.Config,这是上位机的系统配置模块。

Company.Application.Login,这是上位机的用户登录模块。

Company.Application.Initialize,这是上位机的硬件加载模块,当然一些软件方面的加载也可放其中。

Company.Application.Share,这是上位机的共享模块,此模块的作用非常重要,负责各个模块的数据通讯。

Company.Application.Menu,这是上位机的菜单模块,负责菜单生成或管理。

Company.Application.More,这个表示其它模块,根据上位机功能而定。

第三、核心层模块

Company.Core,这是上位机的核心模块,与主业务无关,提供一些基础帮助类,全局类等,比如Json文件的序列化与反序列化,对话框管理器、缓存管理、本地语言包管理、各种helper类型等。

Company.Logger,这是上位机的日志模块,可以采用NLog或Log4net等常用组件,此模块也与主业务无关,被其它所有模块所引用,负责打印程序错误信息或调试信息。

Company.UI,这是上位机关于UI的资源模块,如WPF的模板样式定义、程序所引用的图标、图像、字体库、语言包、转换器,用户控件等。

第四、硬件层模块

Company.Hardware.Camera,这是上位机的工业相机抽象类模块,负责定义一台相机的功能,定义相机的配置参数和操作业务流程(如打开相机、使用相机、关闭相机)。

Company.Hardware.Camera.HIK,表示一台海康工业相机,它继承了相机抽象类,并实现其自身的API调用。

Company.Hardware.Camera.Other,表示其它工业相机,other要换成具体的相机名称,说明可能无限扩展其它厂家或其它型号的相机。

Company.Hardware.ControlCard,这是上位机的控制卡抽象基类,负责定义一张控制卡的功能,但不实现具体的操作流程,只规定操作接口,具体操作由子类实现。

Company.Hardware.ControlCard.DMC2610,比如雷赛运动控制卡DMC2610。

Company.Hardware.ControlCard.ADTech632XE,比如众为兴运动控制卡632XE。

其它的硬件都可以采用上述的命名方式去扩展。

第五、数据库模块

Company.Database.Core,这个模块定义了数据库的操作接口,仓储层的抽象类型 等。

Company.Database.EF,这是采用EF框架去对数据库进行增删改查的模块,其中的类型都继承于Core中的抽象类。

接下来,用一张结构图,演示详细的上位机架构
在这里插入图片描述
上位机开发架构设计高清大图链接地址 :

上位机课程思维导图

https://www.processon.com/view/link/64a10419111c1d7d8a19db04

重庆教主的VIP课程

WPF中关于Prism框架的学习课程

https://edu.51cto.com/course/33880.html

C#+WPF上位机开发课程(模块化与反应式编程)

https://edu.51cto.com/course/34143.html

C#+WPF项目实战MVVM模式开发《超市管理系统》

https://edu.51cto.com/course/33794.html

作者:重庆教主

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

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

相关文章

css图标 | 来自 fontawesome 字体文件的586 个小图标

1. css效果 /*!* Font Awesome 4.4.0 by davegandy - http://fontawesome.io - fontawesome* License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License)*/.fa-glass:before {content:"\f000"} .fa-music:before {content:"\f001"…

小白如何理解MySQL?一文吃透

从本质上来说,MySQL也是一个软件,以Java为例,Java通过JDBC进行MySQL驱动连接后,通过调用“MySQL”的“接口”将SQL语句传给MySQL,并获取返回结果! 连接器 第一步,你会先连接到这个数据库上&…

项目管理:项目计划有哪些不可忽视的作用

为了确保项目在我们的预期范围内完成,编制计划是不可或缺的,它可以帮助项目管理团队进行提前思考、识别和管理任何疏漏和风险。 项目计划进行跟踪中有哪些不可忽视的作用: 1、了解成员的工作情况 分配任务后,项目经理应主动与…

蓝海卓越计费管理系统任意文件读取下载

看了大佬的文章,太牛逼啦,下面是大佬文章原文。 蓝海卓越计费管理系统任意文件读取下载 鹰图语法:web.title"蓝海卓越计费管理系统" 访问url 直接更改url就行了 /download.php?file../../../../../etc/passwd

nginx 配置多域名多站点 Ubuntu

nginx 配置多域名多站点 Ubuntu 一、安装 nginx apt install nginx二、配置文件说明 nginx 的配置文件在 /etc/nginx 目录下,它的默认内容是这样的 root2bd0:/etc/nginx# ll total 72 drwxr-xr-x 8 root root 4096 Jul 31 15:21 ./ drwxr-xr-x 104 root root …

需要暴雨天气安全“指南”的不仅仅是个人

昨日,人民日报官微发布#暴雨天气10个安全指南#,从居家防范、行车安全、户外出行、遇灾求救、次生灾害、防疫防病等方面给出了针对暴雨的安全建议。 以上10条指南是主要面向个人的建议,而在城市水利基础设施的运营和维护上,需要一些…

leetcode 860. 柠檬水找零

2023.8.1 简单的一个思路就是建一个大小为3的数组change ,用于存储剩余的零钱,然后遍历账单,每次找零钱的时候判断一下是否有足够的零钱,不够的话直接返回false。 能坚持到结束遍历则返回true。 代码如下: class Solu…

【视觉SLAM入门】5.1. 特征提取和匹配--FAST,ORB(关键点描述子),2D-2D对极几何,本质矩阵,单应矩阵,三角测量,三角化矛盾

"不言而善应" 0. 基础知识1. 特征提取和匹配1.1 FAST关键点1.2 ORB的关键点--改进FAST1.3 ORB的描述子--BRIEF1.4 总结 2. 对极几何,对极约束2.1 本质矩阵(对极约束)2.1.1 求解本质矩阵2.1.2 恢复相机运动 R , t R,t R,…

【React】搭建React项目

最近自己在尝试搭建react项目,其实react项目搭建没有想象中的那么复杂,我们只需要使用一个命令把React架子搭建好,其他的依赖可以根据具体的需求去安装,比如AntDesignMobile的UI框架,执行npm install antd-mobile --sa…

无涯教程-Lua - 变量声明

变量的名称可以由字母,数字和下划线字符组成。它必须以字母或下划线开头,由于Lua区分大小写,因此大写和小写字母是不同的。 在Lua中,尽管无涯教程没有变量数据类型,但是根据变量的范围有三种类型。 全局变量(Global) …

[Linux]详解环境基础开发工具的使用

[Linux]环境基础开发工具的使用 文章目录 [Linux]环境基础开发工具的使用0. 前言1. Linux 软件包管理器 yumyum介绍yum的使用yum源 2. Linux编辑器-vimvim介绍vim基本模式底行模式下的命令汇总命令模式下的命令汇总vim简单配置 3. Linux编译器gcc/g4. Linux项目自动化构建工具-…

IO进程线程第四天(8.1)opendir,closedir,readdir

作业1&#xff1a; 从终端获取一个文件的路径以及名字。 若该文件是目录文件&#xff0c;则将该文件下的所有文件的属性显示到终端&#xff0c;类似ls -l该文件夹 若该文件不是目录文件&#xff0c;则显示该文件的属性到终端上&#xff0c;类似ls -l这单个文件 #include<…

Linux|ubuntu下运行python

参考&#xff1a;ubuntu系统下切换python版本的方法 文章目录 python版本问题查看ubuntu下的所有python版本通过apt-get install可以安装不同版本python查看python版本号更新update-alternatives替代列表查看update-alternatives下的python版本切换python版本删除python版本 p…

工作记录------单元测试(持续更新)

工作记录------单元测试 之前的工作中从来没有写过单元测试&#xff0c;新入职公司要求写单元测试&#xff0c; 个人觉得&#xff0c;作为程序员单元测试还是必须会写的 于此记录一下首次编写单元测试的过程。 首先引入单元测试相关的依赖 <dependency><groupId>…

分布式开源监控Zabbix实战

Zabbix作为一个分布式开源监控软件&#xff0c;在传统的监控领域有着先天的优势&#xff0c;具备灵活的数据采集、自定义的告警策略、丰富的图表展示以及高可用性和扩展性。本文简要介绍Zabbix的特性、整体架构和工作流程&#xff0c;以及安装部署的过程&#xff0c;并结合实战…

【爬虫逆向案例】某易云音乐(评论)js逆向—— params、encSecKey解密

声明&#xff1a;本文只作学习研究&#xff0c;禁止用于非法用途&#xff0c;否则后果自负&#xff0c;如有侵权&#xff0c;请告知删除&#xff0c;谢谢&#xff01; 【爬虫逆向案例】某易云音乐&#xff08;评论&#xff09;js逆向—— params、encSecKey解密 1、前言2、行动…

EXCEL, 用if({1,0,0} ...) 实现把给定的区域,输出为任意你想要的矩阵,数组区域!

目录 1 原材料&#xff1a;这样的一个区域 工具 if({1,0,0}) 数组公式 1.1 原始数据 1.2 原理 if(0/1,t-value,f-value)---变形--->if({},range1,range2) 1.2.1 if(0/1,t-value,f-value)---变形--->if({},range1,range2) 1.2.2 原理1&#xff1a; if 数组原理&#…

汽车后视镜反射率检测系统

随着社会的快速发展和物质生活的提供&#xff0c;机动车越来越普及&#xff0c;道路行车安全日益重要。为了保障机动车辆和行人的安全&#xff0c;在行车时不断观察后方和两侧的图像尤为重要。机动车后视镜通过反射镜面可以提供在规定视野内后方和两侧的图像&#xff0c;从而提…

【云原生】Kubernetes中deployment是什么?

目录 Deployments 更新 Deployment 回滚 Deployment 缩放 Deployment Deployment 状态 清理策略 金丝雀部署 编写 Deployment 规约 Deployments 一个 Deployment 为 Pod 和 ReplicaSet 提供声明式的更新能力。 你负责描述 Deployment 中的 目标状态&#xff0c;而 De…

Red Hat 安装JDK与IntelliJ IDEA

目录 前言 Red Hat 安装 JDK 1、更新软件包列表 2、安装OpenJDK 3、验证安装 Red Hat 安装IntelliJ IDEA 1、下载 IntelliJ IDEA 2、解压缩 IntelliJ IDEA 安装包 3、移动 IntelliJ IDEA 到安装目录 4、启动 IntelliJ IDEA 前言 YUM是基于Red Hat的Linux发行版的一个…