C#知识点-16(计算器插件开发、事件、递归、XML)

news2025/1/10 18:23:41

计算器插件开发

1、Calculator.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace Calculator_DLL
{
    //用来明确所有插件开发人员的开发规范
    public abstract class Calculator
    {
        public int NumberOne { get;set; }
        public int NumberTwo { get;set; }
        public abstract string Oper { get; }//抽象属性
        public Calculator(int n1,int n2)
        {
            this.NumberOne = n1;
            this.NumberTwo = n2;
        }
        //提供一个计算的方法,具体的计算公式,留给插件开发人员
        public abstract int GetResult();
    }
}

2、Calculator_Add.cs

using Calculator_DLL;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace Calculator_DLL_Add
{
    public class Calculator_Add : Calculator
    {
        public Calculator_Add(int n1, int n2) : base(n1, n2)
        {
        }

        public override string Oper { get { return "+"; } }

        public override int GetResult()
        {
            return this.NumberOne + this.NumberTwo;
        }
    }
}

3、Calculator_Sub.cs

using Calculator_DLL;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace Calculator_DLL_Sub
{
    public class Calculator_Sub : Calculator
    {
        public Calculator_Sub(int n1, int n2) : base(n1, n2)
        {
        }

        public override string Oper { get { return "-"; } }

        public override int GetResult()
        {
            return this.NumberOne-this.NumberTwo;
        }
    }
}

4、Calculator_Factory.cs

using Calculator_DLL;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;

namespace Calculator_Factory_DLL
{
    /// <summary>
    /// 计算器插件的工厂,根据用户的选择,返回对应的计算对象(插件对象)
    /// </summary>
    public class Calculator_Factory
    {
        /// <summary>
        /// 根据用户选择的操作符,创建对应的插件对象(是用父类屏蔽了所有子类插件对象)
        /// </summary>
        /// <param name="oper">用户选择的操作符</param>
        /// <param name="n1">数字1</param>
        /// <param name="n2">数字2</param>
        /// <returns>对应的计算插件对象</returns>
        public static Calculator GetCalculator(string oper,int n1,int n2)
        {
            Calculator cal = null;
            //Assembly:表示程序集 GetExecutingAssembly():获取包含当前执行的代码的程序集  Location:位置,也就是路径
            string path = Assembly.GetExecutingAssembly().Location;
            //GetDirectoryName():返回指定路径字符串的目录信息
            path = Path.GetDirectoryName(path);//DLL程序集的上一个目录,也就是Debug目录
            path = Path.Combine(path, "Plug-in-Components");
            //读取文件夹下的所有插件(Add、Sub)
            string[] files = Directory.GetFiles(path);
            //遍历所有类库文件,从里面找到我们规范的类型(类型要求:继承了Calculator,并且重写了其中的抽象成员,并且不是抽象的)
            foreach (var file in files)
            {
                //LoadFile():加载指定路径上的程序集文件的内容
                Assembly ass = Assembly.LoadFile(file);
                //GetExportedTypes():获取程序集中定义的公共类型
                Type[] types = ass.GetExportedTypes();
                //判断类型是否为我们需要的类型
                foreach (var type in types)
                {
                    //开始筛选
                    //IsAssignableFrom():确定指定类型的实例是否能分配给当前实例(看有没有继承关系)
                    //IsAbstract:判断当前类型是否是抽象的
                    if (typeof(Calculator).IsAssignableFrom(type)&&!type.IsAbstract)
                    {
                        //创建type的对象
                        object o =Activator.CreateInstance(type,n1,n2);
                        cal = o as Calculator;
                        //如果计算对象的Oper属性==用户传入的Oper值
                        if (cal.Oper==oper)
                        {
                            return cal;
                        }
                    }
                }
            }
            return cal;
        }
    }
}

5、Form1.cs

using Calculator_DLL;
using Calculator_Factory_DLL;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace 插件记事本
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            //1、读取配置文件,创建对应的按钮对象
            string path = Assembly.GetExecutingAssembly().Location;
            path=Path.GetDirectoryName(path);
            path = Path.Combine(path, "CalculatorConfig.txt");
            //2、读取配置文件
            string[] operLines = File.ReadAllLines(path);
            int x = 100;
            //3、创建按钮对象
            for (int i = 0; i < operLines.Length; i++)
            {
                Button btn = new Button();
                btn.Text = operLines[i];
                btn.Size = new Size(75, 23);
                btn.Location = new Point(326 + i * x, 257);
                btn.Click += Btn_Click;
                this.Controls.Add(btn);
            }
        }

        private void Btn_Click(object sender, EventArgs e)
        {
            Button btn = sender as Button;
            string oper = btn.Text;
            Calculator cal = Calculator_Factory.GetCalculator(oper,int.Parse(txtNumberOne.Text),int.Parse(txtNumberTwo.Text));
            if (cal != null)
            {
                lblResult.Text = cal.GetResult().ToString();
            }
        }
    }
}

事件

概念:事件,就是一个类型安全的委托(事件是安全的,因为在类的外部,事件只能被赋值,而不能被调用)
事件的三个重要组成部分:注册事件、触发事件、响应事件

递归

概念:在方法中,自己调用自己。必须在满足某个条件的时候,退出递归

递归查找所有的文件夹和文件

        private void Form1_Load(object sender, EventArgs e)
        {
            //递归加载指定目录下,所有的文件夹和文件
            string path = @"C:\Windows\Microsoft.NET\Framework64\v4.0.30319";
            LoadData(path, treeView2.Nodes);
        }
        private void LoadData(string path, TreeNodeCollection nodes)
        {
            //1、获取该路径下所有的文件夹
            string[] dirs = Directory.GetDirectories(path);
            //1.1 把所有的子目录的路径,添加到Treeview上
            foreach (var item in dirs)
            {
                //tn就是根节点
                TreeNode tn = nodes.Add(Path.GetFileName(item));
                LoadData(item, tn.Nodes);
            }
            //2、获取该路径下所有的文件
            string[] files = Directory.GetFiles(path);
            foreach (var item in files)
            {
                nodes.Add(Path.GetFileName(item));
            }
        }

XML

XML:可扩展的标记语言

通过代码创建XML文档

    internal class Program
    {
        static void Main(string[] args)
        {
            //1、在内存中,创建XML文档对象
            XmlDocument xml = new XmlDocument();
            //2、创建文档声明
            XmlDeclaration dec = xml.CreateXmlDeclaration("1.0", "utf-8", null);
            //2.1 把创建的文档声明,添加到xml文档中
            xml.AppendChild(dec);
            //3、创建根节点
            XmlElement books = xml.CreateElement("Books");
            //3.1 把创建的根节点,添加到xml文档中
            xml.AppendChild(books);

            //4.1、给根节点,添加对应的属性
            XmlElement book1 = xml.CreateElement("Book");
            //Attribute :特性  [Serilizeble]  Property:属性
            XmlAttribute book1_id = xml.CreateAttribute("ID");
            //给属性赋值
            book1_id.Value = "00001";
            //把属性跟Book标签绑定到一起
            book1.Attributes.Append(book1_id);
            books.AppendChild(book1);
            //4.2 给根节点,添加对应的子节点
            XmlElement book1_name = xml.CreateElement("Name");
            //给子节点赋值
            book1_name.InnerText = "金瓶梅";
            book1.AppendChild(book1_name);


            //4.2 给根节点,添加对应的子节点
            XmlElement book1_author = xml.CreateElement("Author");
            //给子节点赋值
            book1_author.InnerText = "西门大官人";
            book1.AppendChild(book1_author);

            //4.2 给根节点,添加对应的子节点
            XmlElement book1_price = xml.CreateElement("Price");
            //给子节点赋值
            book1_price.InnerText = "10";
            book1.AppendChild(book1_price);

            //4.1、给根节点,添加对应的属性
            XmlElement book2 = xml.CreateElement("Book");
            //Attribute :特性  [Serilizeble]  Property:属性
            XmlAttribute book2_id = xml.CreateAttribute("ID");
            //给属性赋值
            book2_id.Value = "00002";
            //把属性跟Book标签绑定到一起
            book2.Attributes.Append(book2_id);
            books.AppendChild(book2);
            //4.2 给根节点,添加对应的子节点
            XmlElement book2_name = xml.CreateElement("Name");
            //给子节点赋值
            book2_name.InnerText = "水浒传";
            book2.AppendChild(book2_name);


            //4.2 给根节点,添加对应的子节点
            XmlElement book2_author = xml.CreateElement("Author");
            //给子节点赋值
            book2_author.InnerText = "潘金莲";
            book2.AppendChild(book2_author);

            //4.2 给根节点,添加对应的子节点
            XmlElement book2_price = xml.CreateElement("Price");
            //给子节点赋值
            book2_price.InnerText = "20";
            book2.AppendChild(book2_price);



            xml.Save("Books.xml");
            Console.WriteLine("保存OK");
            Console.ReadKey();
        }
    }

对XML增删改查

    internal class Program
    {
        static void Main(string[] args)
        {
            List<Person> listPerson = new List<Person>();
            listPerson.AddRange
                (new Person[] {
                new Person() {Name= "张三",Age=19,Gender='男' },
                new Person() {Name= "李四",Age=20,Gender='女' },
                new Person() {Name= "王五",Age=21,Gender='中' }});

            //把集合中的数据,写入到xml文档中
            //1、创建文档对象
            XmlDocument doc = new XmlDocument();
            //2、创建文档声明
            XmlDeclaration dec = doc.CreateXmlDeclaration("1.0", "utf-8", null);
            doc.AppendChild(dec);
            //3、创建根节点
            XmlElement xmlPersons = doc.CreateElement("Persons");
            doc.AppendChild(xmlPersons);
            int i = 0;
            //4、遍历泛型集合,把集合中的Person对象,以元素的形式,添加到XML文档中
            foreach (var item in listPerson)
            {
                i++;
                //4.1 每一个Person对象,都是一个节点
                XmlElement person = doc.CreateElement("Person");
                XmlAttribute att = doc.CreateAttribute("ID");
                att.Value = i.ToString();
                person.Attributes.Append(att);
                //4.2 把创建的子节点,添加到根节点上
                xmlPersons.AppendChild(person);
                //4.3 给子节点,添加对应的Name、Age、Gender
                XmlElement name = doc.CreateElement("Name");
                name.InnerText = item.Name;
                //4.4 把子节点Name,添加到根节点Person上
                person.AppendChild(name);

                //4.3 给子节点,添加对应的Name、Age、Gender
                XmlElement age = doc.CreateElement("Age");
                age.InnerText = item.Age.ToString();
                //4.4 把子节点Name,添加到根节点Person上
                person.AppendChild(age);

                //4.3 给子节点,添加对应的Name、Age、Gender
                XmlElement gender = doc.CreateElement("Gender");
                gender.InnerText = item.Gender.ToString();
                //4.4 把子节点Name,添加到根节点Person上
                person.AppendChild(gender);
            }


            doc.Save("Person.xml");
            Console.WriteLine("保存成功");
            Console.ReadKey();


        }
    }
    class Person
    {
        public string Name { get; set; }

        public int Age { get; set; }

        public char Gender { get; set; }
    }

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

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

相关文章

2024作品集设计指南:3个你需要知道的趋势!

如果你想在面试中获得额外的分数&#xff0c;你的首要任务是仔细准备一个个人作品集。作品集是展示设计师个人能力的载体。一个优秀的作品集不仅可以向面试官展示你的设计技巧&#xff0c;还可以通过将作品集与设计趋势“融合”来体现你对市场的关注。“设计技能市场思维”的作…

Linux环境下查看磁盘层级占用空间的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

Java基于SpringBoot的社区医院信息管理系统

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

iMazing3终极iPhone数据设备管理软件

iMazing是一款功能丰富的iOS设备管理软件&#xff0c;具备多种实用功能&#xff0c;以下是它的主要功能的详细介绍&#xff1a; iMazing3Mac-最新绿色安装包下载如下&#xff1a; https://wm.makeding.com/iclk/?zoneid49816 iMazing3Win-最新绿色安装包下载如下&#xff1…

2024年雾锁王国Enshrouded服务器搭建方法,最新教程

阿里云雾锁王国服务器搭建教程是基于计算巢服务&#xff0c;3分钟即可成功创建Enshrouded游戏服务器&#xff0c;阿里云8核32G雾锁王国专用游戏服务器90元1个月、271元3个月&#xff0c;阿里云服务器网aliyunfuwuqi.com亲自整理雾锁王国服务器详细搭建教程&#xff1a; 一、前…

解决IDEA搜不到插件

File -> Settings -> Plugins https://plugins.jetbrains.com/ 完成以上操作即可搜到插件

SMT贴片加工厂需要哪些加工资料

SMT贴片加工中在评估报价的时候需要给到SMT贴片加工厂以下资料&#xff0c;以便工程师和采购进行工艺和报价评估。 在SMT加工中如果需要供应商提供一站式的加工服务&#xff0c;那么在前期就需要更频繁的沟通和配合&#xff0c;包工包料服务是需要PCB制板资料和制板说明、BOM清…

win32汇编定时器继续学习2

前面学习了win32汇编定时器&#xff0c;还非常不熟悉&#xff0c;继续熟悉&#xff0c;稍微增加一点功能&#xff1b; .386.model flat,stdcalloption casemap:noneinclude windows.inc include user32.inc includelib user32.lib include kernel32.inc includelib kernel32…

【深入理解设计模式】 工厂设计模式

工厂设计模式 工厂设计模式是一种创建型设计模式&#xff0c;它提供了一种在不指定具体类的情况下创建对象的接口。在工厂设计模式中&#xff0c;我们定义一个创建对象的接口&#xff0c;让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。 工厂设计模式的目…

Docker镜像加速

前言 众所周知&#xff0c;我们常用的一些工具或系统的下载源都是国外的&#xff0c;这就会导致我们在下载一些东西时&#xff0c;会导致下载巨慢或者下载失败的情况&#xff0c;下面便是docker换下载源的教程 镜像加速 下面是几个常用的国内的镜像 科大镜像&#xff1a;ht…

十大基础排序算法

排序算法分类 排序&#xff1a;将一组对象按照某种逻辑顺序重新排列的过程。 按照待排序数据的规模分为&#xff1a; 内部排序&#xff1a;数据量不大&#xff0c;全部存在内存中&#xff1b;外部排序&#xff1a;数据量很大&#xff0c;无法一次性全部存在内存中&#xff0c;…

基于SpringBoot + Layui的社区物业管理系统

项目介绍 社区物业管理系统是基于java编程语言&#xff0c;springboot框架&#xff0c;idea工具&#xff0c;mysql数据库进行开发&#xff0c;本系统分为业主和管理员两个角色&#xff0c;业主可以登陆系统&#xff0c;查看车位费用信息&#xff0c;查看物业费用信息&#xff0…

【深度学习:TACO 数据集】探索 TACO 数据集【模型训练】

【深度学习&#xff1a;TACO 数据集】探索 TACO 数据集【模型训练】 介绍为什么选择以数据为中心的人工智能&#xff1f;上次我们学到了什么&#xff1f;问题关于数据集方法 什么是“对象注释质量”指标&#xff1f;第一次迭代&#xff1a;修复标签错误分析重新贴标签模型再训练…

云安全:领先攻击者一步

基于SaaS的错误配置将导致安全事件升级&#xff1a;基于SaaS的应用的受欢迎程度呈指数级增长。然而&#xff0c;相对容易的采用也增加了攻击面。SaaS错误配置安全相关事件的数量可能高达63%&#xff0c;与17%的安全事件相比&#xff0c;这个数字显得巨大。 云安全现在在每项重大…

深入解析SDRAM:从工作原理到实际应用

深入解析SDRAM&#xff1a;从工作原理到实际应用 在众多内存技术中&#xff0c;同步动态随机访问存储器&#xff08;SDRAM&#xff09;因其出色的性能和广泛的应用而备受关注。本文将从SDRAM的工作原理入手&#xff0c;探讨其性能优化策略和在现代电子设备中的应用。 SDRAM工作…

实战打靶集锦-025-HackInOS

文章目录 1. 主机发现2. 端口扫描3. 服务枚举4. 服务探查5. 提权5.1 枚举系统信息5.2 探索一下passwd5.3 枚举可执行文件5.4 查看capabilities位5.5 目录探索5.6 枚举定时任务5.7 Linpeas提权 靶机地址&#xff1a;https://download.vulnhub.com/hackinos/HackInOS.ova 1. 主机…

Nignx的搭建与核心配置

目录 一、Nginx是什么&#xff1f; 1、Nginx概述 2、Nginx模块与作用 3、Nginx三大作用&#xff1a;反向代理&#xff0c;负载均衡&#xff0c;动静分离 nginx七层负载均衡调度算法&#xff08;六种&#xff09; 1、轮询&#xff08;默认调度算法&#xff09; 2、加权轮…

vue如何动态加载显示本地图片资源

在实际开发中&#xff0c;根据某一个变量动态展示图片的情况有很多。实现方法分打包构建工具的差异而不同。 1、webpack的项目 require引入图片资源 2、vite的项目 new URL(url,base).href 疑问解答&#xff1a;为什么vite项目不可以用require&#xff1f; 原因在于&#xf…

BT709 YUV与RGB 色彩空间的转换(处理色差问题)

BT709 YUV与RGB 色彩空间的转换 主要渲染能够更加清新没有色差 CIE xyz色彩系统&#xff0c; 480 460这些数字表示波长&#xff0c;越靠近边缘饱和度越高 由于 XYZ 三基色所合成的色光是由他们的比值所决定的&#xff0c;可以设&#xff1a; ⻢蹄形的左右两边的轮廓线代表了…

UE蓝图 返回结果(FunctionResult)节点和源码

系列文章目录 UE蓝图 Get节点和源码 UE蓝图 Set节点和源码 UE蓝图 Cast节点和源码 UE蓝图 分支(Branch)节点和源码 UE蓝图 入口(FunctionEntry)节点和源码 UE蓝图 返回结果(FunctionResult)节点和源码 文章目录 系列文章目录一、FunctionResult节点功能二、FunctionResult节点…