面向对象五大设计原则

最近在看七牛云许式伟的架构课, 重温了面向对象五大设计原则(SOLID),扣理论文字找出处。(当然许老板是不可能深聊这么低级的内容,🤡)

注意区分设计原则和设计模式。
设计原则更为抽象和泛化;
设计模式也是抽象或泛化的良好实践,但是它们提供了更具体和实用的底层建议。

面向对象5大设计原则
Single Responsiblity Principle单一职责原则
Open/Closed Principle开闭原则
Likov Substitution Principle里斯替代原则
Interface Segregation Principle接口隔离原则
Dependency inversion依赖倒置原则

单一职责原则

只能有一个让组件或类发生改变的原因;或者说每个组件或类专注于单一功能,解决特定问题。

there should never be more than one reason for a class to change. A class should be focused on a single functionality, address a specific concern.

开闭原则

对扩展开放, 对修改封闭。

扩展类的几种方式:

  • • 从类继承

  • • 类中重写同名行为

  • • 扩展类的某些行为

一般我们通过继承或者实现接口来实践开闭原则。

class Person{public int age;public string name;public Person(string name, int age){this.name = name;this.age = age;}public virtual void SayHallo(){Console.WriteLine("我是{0},今年{1}", name, age);}}class Student : Person{public string major;public Student(string name, int age, string major) : base(name, age){this.major = major;}public override void SayHallo()   //子类中override重写,实现虚方法{Console.WriteLine("我是{0},今年{1},正在学习{2}", name, age, major);}}class Program{static void Main(string[] args){Person trevor1 = new Person("Trevor", 18);trevor1.SayHallo();Student trevor2 = new Student("Trevor", 18,"C#");trevor2.SayHallo();}}output:
我是Trevor,今年18
我是Trevor,今年18,正在学习C#

里氏替代原则

在父子类生态中,在父类出现的地方,可以用子类对象替换父类对象,同时不改变程序的功能和正确性。

。。 乍一看,这不是理所当然吗?

为啥单独拎出来鞭尸,鞭策。

比如上例我们使用

Person trevor1 = new Student("trevor",18,"C#")  // 子类对象替换父类对象trevor1.SayHello();

利用多态正确表达了含义。


但是某些情况下滥用继承,却不一定保证程序的正确性,会对使用者造成误解。

比如下面经典的[矩形-正方形求面积]反例:

public class Rectangle
{// 分别设置宽高public virtual double Width {get;set;}public virtual double Height {get;set;}public virtual void Area(){Console.WriteLine("面积是:" + Width * Height);}
}public class Square : Rectangle
{public override double Width {// get;set   //  因为是正方形,想当然重设了宽=高{base.Width= value;base.Height= value;}}public override double Height{//  get;set  //  因为是正方形,想当然重设了宽=高{base.Width = value;base.Height = value;}}public override void Area(){Console.WriteLine("面积是:" + Width * Width);} 
}public  class Program
{public static void Main(){Rectangle s = new Rectangle();s.Width = 2;          s.Height = 3;         s.Area();}
}output:
面积是:6

但是如果你[使用子类对象去替换父类对象]:

Rectangle s2 = new Square();s2.Width = 2;          s2.Height = 3;         s2.Area();output:
面积是:9

Get到了吗?我们不能想当然的认为子类对象就能无损替换父类对象, 根本原因是我们正方形虽然是(is a)矩形,但是我们的重写行为破坏了父类的表达,这是一种继承的误用。

里氏替代原则就是约束你在继承(is a)的时候注意到这个现象,并提醒你规避这个问题。

这个时候,不应该重写父类的SetWight方法, 而应该扩展新的方法SetLength。

接口隔离 

将胖接口修改为多个小接口,调用接口的代码应该比实现接口的代码更依赖于接口。

why:如果一个类实现了胖接口的所有方法(部分方法在某次调用时并不需要),那么在该次调用时我们就会发现此时出现了(部分并不需要的方法),而并没有机制告诉我们现在不应该使用这部分方法。

how:
避免胖接口,不要实现违反单一职责原则的接口。可以根据实际多职责划分为多接口,某个类实现多接口后, 在调用时以特定接口指代对象,这样这个对象只能体现特定接口的方法,以此体现接口隔离。

public interface IA{void getA();}interface IB{void getB();}public class Test : IA, IB{public string Field { get; set; }public void getA(){throw new NotImplementedException();}public void getB(){throw new NotImplementedException();}}class Program{static void Main(string[] args){Console.WriteLine("Hello World!");IA a = new Test();a.getA();       //  在这个调用处只能看到接口IA的方法, 接口隔离}}

依赖倒置原则

实现依赖于抽象, 抽象不依赖于细节。

Q:这个原则我其实一开始没能理解什么叫“倒置”?

A: 但有了一点开发经验后开始有点心得了。

痛点:面向过程的开发,上层调用下层,上层依赖于下层。当下层变动时上层也要跟着变动,导致模块复用度降低,维护成本增高。

5d1745f7cbde8e00fa429ed55c6863c2.png

提炼痛点:含有高层策略的模块,如AutoSystem模块,依赖于它所控制的低层的负责具体细节的模块。

思路:找到一种方法使AutoSystem模块独立于它所控制的具体细节,那么我们就可以自由地复用AutoSystem了;同时让底层汽车厂也依赖抽象,受抽象驱动,这就形成一种“倒置”。

cd15cb8114597691f301d9cc5698ebcb.png

所以依赖倒置原则有两个关键体现:

  1. ①  高层次模块不应该依赖于低层实现,而都应该依赖于抽象;

    这在上图:AutoSystem和Car都依赖于抽象接口ICar
  2. ②  抽象不应该依赖于具体实现,具体实现应该依赖于抽象。

    第2点与第1点不是重复的,这一点意味着细节实现是受抽象驱动,这也是“倒置”的由来, 这一点是通过接口叫ICar而不是IAutoSystem来体现。

面相对象五大设计原则SOLID,是指导思想,不贯彻这5大设计原则也能让程序跑起来,但是可能就会出现阅读性、维护性、正确性问题。

本人会不时修正理解、更正错误,请适时移步左下角永久更新地址;也请看客大胆斧正。

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

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

相关文章

谷歌F12调试公众号时,让鼠标显示出来

yi 环境介绍: win10 , 谷歌浏览器yii 概述: 在项目中,需要调试公众号,本地环境搭好之后,在谷歌浏览时,发现移动到公众号区域,鼠标居然不见了,这让我怎么操作?各种操作可谓是日了狗了,非常麻烦yiii 调试时鼠标不见的解决办法: 网上各种说法众说纷纭,这里,我给出本人认为最恰当简…

利用bootstrap插件设置时间

$("#"id_rand" .shijian-input").each(function () { $(this).datetimepicker({ lang:"ch", //语言选择中文 注:旧版本 新版方法:$.datetimepicker.setLocale(ch); format: "hh : ii", /…

C# 编写的 64位操作系统 -MOOS

MOOSMOOS ( My Own Operating System )是一个使用.NET Native AOT技术编译的C# 64位操作系统。项目地址:https://github.com/nifanfa/MOOS编译关于编译MOOS的信息,请阅读 编译维基页面:https://github.com/nifanfa/MOOS/wiki/。编译要求VMwar…

JAVA语言基础-面向对象(IO:IO字符流、递归)

2019独角兽企业重金招聘Python工程师标准>>> 21.01_IO流(字符流FileReader) 1.字符流是什么 字符流是可以直接读写字符的IO流字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要写出字符, 需要把字符转为字节再写出.2.FileReader FileReader类的read()方法…

windows下, nginx 提示错误 No input file specified

一 环境介绍: win10, LNMP 二 错误描述: 访问网站时,提示"No input file specified"错误. 排错阶段: 1. 查看nginx access日志 (access.log) 发现提示404 错误 2. 分析原因: 这时,在同目录下创建一个txt文件,访问就可以正常输出了 这说明 现在nginx 访问php 没…

Ubuntu20.04+docker+jenkins+飞书实现自动化发布

一、从0-1一点一滴实现如何本地提交代码到gitlab然后实现前后端自动发布1.更新apt包索引sudo apt-get update2.安装必备的软件包以允许apt通过https使用存储库sudo apt-get install ca-certificates curl gnupg lsb-release3.添加Docker官方版本的GPG密钥sudo mkdir -p /etc/ap…

一个Demo让你掌握Android所有控件

一个Demo让你掌握Android所有控件 原文:一个Demo让你掌握Android所有控件本文是转载收藏,侵删,出处:"安卓巴士" 下面给出实现各个组件的源代码: 1.下拉框实现--Spinner [java] view plaincopyprint?package com.cellcom; import java.util.ArrayList;…

九妹带你走向 架构师

迈向系统架构师编者按:系统架构师是许多程序员的梦想职业。今天的你也许已经掌握了各种开发工具,并且能够使用各种平台进行开发,但作为一个架构师的要求,也许还有很长的道路。邢波涛先生在LAMP架构上的造诣,让我邀请他…

WPF 使用 DrawingContext 绘制温度计

WPF 使用 DrawingContext 绘制温度计控件名:Thermometer作者: WPFDevelopersOrg原文链接: https://github.com/WPFDevelopersOrg/WPFDevelopers框架使用大于等于.NET40;Visual Studio 2022;项目使用 MIT 开源许可协议&#xff…

装win10系统

一、使用U盘介质安装win10系统(官方方式) 官方安装工具下载地址:https://www.microsoft.com/zh-cn/software-download/windows10 1、进入官方安装工具下载页面,点击立即下载工具,下载安装工具。2、下载完成后&#xff…

C#构造函数、操作符重载以及自定义类型转换

构造器 构造器(构造函数)是将类型的实例初始化的特殊方法。构造器可分为实例构造器和类型构造器,本节将详细介绍有关内容。 实例构造器 顾名思义,实例构造器的作用就是对类型的实例进行初始化。如果类没有显示定义任何构造器&…

「Dotnet 工具箱」 自动生成并绑定 Https 证书

这里是 Dotnet 工具箱,定期分享 Dotnet 有趣,有用的工具,不要忘记关注。介绍LettuceEncrypt 是一个使用 C# 开发的免费的工具,它和证书颁发机构 (CA)集成,比如 Lets Encrypt,它使用了…

1115: 零起点学算法22——华氏摄氏温度转换

1115: 零起点学算法22——华氏摄氏温度转换 Time Limit: 1 Sec Memory Limit: 64 MB 64bit IO Format: %lldSubmitted: 3522 Accepted: 1456[Submit][Status][Web Board]Description 输入一个华氏温度,根据公式C(5/9)(F-32)计算对应的摄氏温度。 Input 输入一个…

Navicat Premium 12 的安装破解

Navicat 这款软件可以说 是数据库可视化操作的神器, 有绿色的 (最原始版本, 好像现在已经不维护了) , 有金色的 (改良收费版 ) , 还有彩色的 (最新版) , 这里 , 推荐使用彩色版 (也就是截止目前最新的版本 12.0.27). 操作的话, 感觉相比于小绿和小金有很大改进 , 很棒 , 在此给…

Vuejs——组件——slot内容分发

2019独角兽企业重金招聘Python工程师标准>>> ①概述: 简单来说,假如父组件需要在子组件内放一些DOM,那么这些DOM是显示、不显示、在哪个地方显示、如何显示,就是slot分发负责的活。 ②默认情况下 父组件在子组件内套的…

turtle库基础练习

画一组同切圆 import turtleturtle.shape(turtle)turtle.circle(10) turtle.circle(20) turtle.circle(30) turtle.circle(40) turtle.circle(50) turtle.circle(60) turtle.circle(70) turtle.circle(80)turtle.hideturtle() turtle.done() 画一组同心圆 import turtleturtle.…

检查你的项目的引用包依赖关系

2019独角兽企业重金招聘Python工程师标准>>> 随着着开发的进展,你的项目越来越大,引用的第三方包越来越多,但如何查看都依赖了哪些包,甚至传递依赖又是怎样? 首先解决这个问题的前提,你的项目需要是maven项目,然后可以做如下设置: 选中项目,右键->ru…

git 项目操作

1 创建本地仓库,克隆远程项目代码到本地仓库2. 当我们在本地写了一些代码之后 , 查看本地仓库状态3. 提交改变到待提交区 git add .4. 提交代码到待推送区 git commit -m "新建项目kuman"5. 将本地代码推送到远程代码仓库 git push origin master:nanle 注: 将本地m…

(二)SpringBoot功能

web开发 spring boot web开发非常的简单,其中包括常用的json输出、filters、property、log等 json 接口开发 在以前的spring 开发的时候需要我们提供json接口的时候需要做那些配置呢 就这样我们会经常由于配置错误,导致406错误等等,spring bo…

----斐波那契数列---eval函数----类递归思想 栈 进出 思想

------------ 斐波那契 数列 ---------------【1&#xff0c;1,2,3,5,8,13,21,34&#xff0c;...】 1 列表方法实现 # l[1,1] # # # while len(l)<20: # # l.append(l[-1]l[-2]) # # print(l) # # while len(l)!4: # l.append(l[-1]l[-2]) # print(l) # 2 …