设计模式简介


设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。是一套被反复使用的、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了重用代码、让代码更容易被他人理解、保证代码可靠性。

设计模式由来


由GOF(四人帮)出版的一本名为《设计模式 - 可复用的面向对象软件元素》的书提出的设计模式的概念。该书基于以下的面向对象设计原则

  • 对接口编程而不是对实现编程。
  • 优先使用对象组合而不是继承。

设计模式分类


常说的23种设计模式是基于《设计模式 - 可复用的面向对象软件元素》提出的,这些模式可以分:创建型模式,结构性模式,行为型模式。

事实上还有一些关于J2EE设计模式,并发型模式,线程池模式等等。

创建型模式(5种):

描述:指提供另外一种创建对象的同时隐藏创建逻辑的方式,不是通过使用new运算符直接实例化对象

包含:工厂方法模式、抽象工厂模式、单例模式、建造者模式、原型模式

结构型模式(7种):

描述:关注类和对象的组合。继承的概念被用来组合接口和定义组合对象获得新功能的方式

包含:适配器模式、装饰器模式、代理模式、外观模式、桥接模式、组合模式、享元模式

行为型模式(11种):

描述:关注对象之间的通信。

包含:策略模式、模板方法模式、观察者模式、迭代子模式、责任链模式、命令模式、备忘录模式、状态模式、访问者模式、中介者模式、解释器模式

下面是关于设计模式之间关系的图

设计模式的六大原则


设计模式遵守六大原则,分别是:开闭原则,里式代换原则,依赖倒转原则,接口隔离原则,迪米特法则(最少知道原则),合成复用原则。
也有一种说法,六大原则应该是七大原则,即在上面的基础上新增单一职责原则

开闭原则(Open Close Principle)

定义:一个软件实体应当对扩展开放,对修改关闭。即软件实体应尽量在不修改原有代码的情况下进行扩展。

意思是在设计的时候要考虑这个类是足够好的,一旦写好之后就不再修改,之后如果有新需求,只需要扩展一些类即可。所以这个原则有两个特性:“对扩展开放”,“对修改关闭”。想要达到这种效果,需要使用接口和抽象类

例子

这是一个书店卖书的例子。
书店有书这个抽象的接口类,里面有定义名字,价格,作者。
相比书这个抽象的接口类,有一个具体的实现类,即小说书。小说书实现了书的功能,通过构造器注入小说书的属性。
书店是使用这些书的一个使用者,进行出售。

如果我们现在要将小说书进行低价9折出售,有什么办法?办法有三:

  1. 修改接口IBook,新添加一个getOffPrice(),获取低价打折后的价格。但是修改之后,NovelBook需要新添加一个getOffPrice()方法,而BookStore类使用时也要进行修改,手动添加打折后的价格。所以不可取
  2. 修改NovelBook类的getPrice()方法,返回打折后的价格给用户,但是这有一个缺陷。即书的原价将会被隐藏掉,看到的只会是打折后的价格。
  3. 扩展NovelBook,新添加一个继承与NovelBook的子类OffNovelBook,该类覆写NovelBook的price,返回打折价格。这样如果书店里的购书者想看书的原价,那么我们就IBook book = new NovelBook()给他看原价,如果想看书的打折价格,就IBook book = new OffNovelBook()

好处
  • 可以提高复用性
  • 可以提高可维护性
  • 针对面向对象开发的要求

里氏代换原则(Liskov Substitution Principle)

里氏代换原则是面向对象设计的基本原则之一。 里氏代换原则中说,任何基类可以出现的地方,子类一定可以出现。
里氏代换原则是对“开-闭”原则的具体实现手段之一。实现“开-闭”原则的关键步骤就是抽象化。而基类与子类的继承关系就是抽象化的具体实现,所以里氏代换原则是对实现抽象化的具体步骤的规范。

定义:所有引用基类(父类)的地方必须能透明地使用其子类的对象。

简单的来讲,就是父类能出现的地方,子类也能出现,即必须能够实现IBook book = new NovelBook();和NovelBook book = new NovelBook()。既java继承。

就是因为这种代换的特性,才使继承复合成为可能,才能谈得上扩展开放,修改关闭。

里式代换原则包含4层含义:

  • 子类必须完全实现父类的抽象方法
        此处的父类代指接口或抽象类,尽量把父类设计为抽象类或者接口。如果是普通类,那么子类必须继承父类的所有方法
  • 子类可以有自己特有的方法

  • 当子类的方法重载父类的方法时,方法的前置条件(即方法的形参)要比父类方法的输入参数更宽松,

         这样能保证父类方法一定会被执行。如果你想让子类的方法能够执行,那么你只能覆写父类方法。所以子类中方法的形参必须与父类中被覆写的方法的形参相同(覆写)或者更大(重载)。 
  • 当子类的方法实现父类的抽象方法时,方法的后置条件(即方法的返回值)要比父类更严格。
        所以子类中方法的返回值必须与父类中被覆写的方法的返回值相同(覆写)或者更小(重载)。 

注意:事实上里氏代换原则就是继承!注意项也都是正确使用继承中应该注意的点,只要形参和返回值与父类被覆写的方法相同,那么你将不会遇到违背继承意义的事情,也就不会违背里氏替换原则

依赖倒转原则(Dependence Inversion Principle)

定义:高层模块不应该依赖低层模块,二者都应该依赖其抽象;抽象不应该依赖于细节,细节应当依赖于抽象。换言之,要针对接口编程,而不是针对实现编程。

这个是开闭原则的基础,要求我们在程序代码中传递参数时或在关联关系中,尽量引用层次高的抽象层类,即使用接口和抽象类进行变量类型声明、参数类型声明、方法返回类型声明,以及数据类型的转换等,而不要用具体类来做这些事情。
抽象类和接口定义方法,实现类实现方法。调用时针对接口进行编程,这就是依赖倒转原则,即实现类依赖于接口,而不是接口依赖于实现

注意: 总之,依赖倒置原则就是要我们面向接口编程,就是向上转型,理解了面向接口编程,也就理解了依赖倒置。

例子:

违背依赖倒置原则,不面向接口编程的UML图是这样的


如上图,司机可以开车,但是司机只能开奔驰车,不能开宝马车。如果司机要开宝马车,只能在司机本身上修改。这不符合要求,一个司机既然有了C驾照,那自然是宝马车和奔驰车都能开。

遵守依赖倒置原则,针对接口编程的UML图是这样的

IDriver接口可以理解为C级驾照,只要考了这个驾照就可以开C级车,即ICar接口,也就可以开宝马车和奔驰车。
所以只要司机有了C级驾照,就可以开奔驰车和宝马车了,这就复合我们的要求。


Client属于高层业务逻辑,Driver和Benz,BMW属于底层业务逻辑,client不依赖于Driver和Benz,而依赖于抽象ICar和IDriver。
可能你看到client这个高层模块调用了Benz类和Driver类这两个底层模块。但是你要知道,向上转型是什么意思。zhangSan的引用是IDriver,是一个抽象的接口,其后的操作都是以IDriver这个接口来进行调用操作的,Benz这个类的实现细节对zhangSan来说是屏蔽的。就像你只要能开车,但是不一定需要知道汽车为什么能开,怎么开,是一样的道理。

关于依赖注入

在实现依赖倒转原则时,我们需要针对抽象层编程,而将具体类的对象通过依赖注入(DependencyInjection, DI)的方式注入到其他对象中。
依赖注入是指当一个对象要与其他对象发生依赖关系时,通过抽象来注入所依赖的对象。常用的注入方式有三种,分别是:构造注入,设值注入(Setter注入)和接口注入。

  1. 构造注入

    如:

         public Driver(ICar car){this.car=car;}
    
  2. setter注入

    如:

         public void setCar(ICar car){this.car=car;}
    
  3. 接口注入

    如:

     public interface IDriver{public void driver(ICar car);
    }
    public class Driver implements IDriver{public void driver(ICar car){car.run();}}
    

    因为Driver类实现了IDriver接口,当你IDriver driver = new Driver();实例化对象后,调用driver方法,传入参数。
    这就是一个接口注入。
    你是通过IDriver接口的driver方法将参数注入的,通过接口注入参数,实例化的是Driver类,调用了driver方法。

如何进行依赖倒置(面向接口编程)

如上面所说,程序代码中传递参数时或在关联关系中,尽量引用层次高的抽象层类。

  1. 方法形参针对抽象类进行编程

    如:

         public void driver(ICar car){car.run();}
    
  2. 声明类型时针对抽象类生成引用

    如:

         ICar car = new Benz();
    
  3. 方法返回类型针对抽象类进行返回

    如:

         public ICar getCar(ICar car){return car;}
    
总结:

为什么叫依赖倒置?
因为依赖正置就是实实在在的类的依赖,面向实现编程编程,高层模块依赖于底层模块,底层模块依赖于高层模块。这也是正常人的思维方式,我要开奔驰车就依赖奔驰车。
倒置是什么?
就是颠覆了人对依赖的认知,我们对奔驰车进行抽象化,抽象成车。那么我依赖的是车这个抽象,而不是奔驰车这个实体,奔驰车也不依赖我,依赖的是车这个抽象,这就是高层模块不应该依赖低层模块,二者都应该依赖其抽象。

接口隔离原则(Interface Segregation Principle)

定义:使用多个专门的接口,而不使用单一的总接口,即客户端不应该依赖那些它不需要的接口,类间的依赖关系应该建立在最小的接口上。

即建立单一接口,不要建立臃肿庞大的接口,接口尽量细化,同时接口中的方法尽量少。客户端需要什么接口就提供什么接口,把不需要的接口剔除掉,那就需要对接口进行细化,保证其纯洁性,把一个臃肿的接口变更为多个独立的接口所依赖的原则就是接口隔离原则

例子:

美女的标准是身材好,脸蛋漂亮,气质好。这样一个接口有时候不能定义,所以可以分为身材好,脸蛋漂亮和气质好两个接口,定义了外表美女和气质美女

注意:

  1. 接口尽量小,但是要有限度。对接口进行细化可以提高程序设计灵活性是不挣的事实,但是如果过小,则会造成接口数量过多,使设计复杂化。所以一定要适度。
  2. 为依赖接口的类定制服务,只暴露给调用的类它需要的方法,它不需要的方法则隐藏起来。只有专注地为一个模块提供定制服务,才能建立最小的依赖关系。
  3. 提高内聚,减少对外交互。使接口用最少的方法去完成最多的事情。
    运用接口隔离原则,一定要适度,接口设计的过大或过小都不好。设计接口的时候,只有多花些时间去思考和筹划,才能准确地实践这一原则。

迪米特法则(最少知道原则)(Demeter Principle)

定义:一个对象应该对其他对象有最少的了解。

典型例子:

  • 只同你直接的朋友们通信
  • 不要跟陌生人说话
  • 每一个软件单位对其他的单位都只有最少的了解,这些了解仅局限于那些与本单位密切相关的软件单位

其根本思想,是强调了类之间的松耦合,类之间的耦合越弱,越有利于复用,一个处在弱耦合的类被修改,不会对有关系的类造成影响,也就是说,信息的隐藏促进了软件的复用。

其实就是要求我们在设计的时候,尽量减少两个对象之间直接的交互。如果其中一个对象需要调用另一个对象时,引入一个第三者来转发这个调用

单一职责原则(Single Responsibility Principle, SRP):

定义:一个类只负责一个功能领域中的相应职责,或者可以定义为:就一个类而言,应该只有一个引起它变化的原因。

遵循单一职责原的优点有:
1.可以降低类的复杂度,一个类只负责一项职责,其逻辑肯定要比负责多项职责简单的多;
2.提高类的可读性,提高系统的可维护性;
3.变更引起的风险降低,变更是必然的,如果单一职责原则遵守的好,当修改一个功能时,可以显著降低对其他功能的影响。

所以遵循单一原则,就要遵循按职责划分接口,一个接口或类,只负责一项职责,是实现高内聚、低耦合的指导方针。它是最简单但又最难运用的原则,需要设计人员发现类的不同职责并将其分离,而发现类的多重职责需要设计人员具有较强的分析设计能力和相关实践经验。

例子:

按职责划分,分为用户行为和用户信息,可划分两个接口

入门篇总结:


开闭原则:对扩展开放,对修改关闭。

里氏代换原则:实现子父类互相替换,即继承;

依赖倒转原则:针对接口编程,实现开闭原则的基础;

接口隔离原则:降低耦合度,接口单独设计,互相隔离;

迪米特法则,又称不知道原则:功能模块尽量独立;

单一职责原则:一个类只负责一项职责

合成复用原则:尽量使用聚合,组合,而不是继承;

后期会再次修改。。。。

设计模式01-入门介绍篇相关推荐

  1. 【Java基础教程】(一)入门介绍篇 · 上:快速掌握核心概念,开启Java世界的探索之旅!这篇Java入门宝典助你翱翔~

    Java基础教程之入门介绍 · 上 本节学习目标 1️⃣ Java发展简史 1.1 诞生 1.2 发展 1.3 分支 2️⃣ 特征 3️⃣ 代码执行过程

  2. Ajax入门介绍篇:Ajax开发基础

    五年前,如果不知道 XML,您就是一只无人重视的丑小鸭.十八个月前,Ruby 成了关注的中心,不知道 Ruby 的程序员只能坐冷板凳了.今天,如果想跟上最新的技术时尚,那您的目标就是 Ajax. 但是 ...

  3. 老宇哥带你玩转ESP32:01入门介绍

    接触物联网差不多10年了. 先跟大家聊聊,老宇哥11年进入大学,大一就在实验室开始了电子研发,记得那时候师兄介绍我买了一块郭天祥老师的51开发板,还有配套的一本书,虽然从小非常喜欢电子,经常折腾,有一 ...

  4. lucene入门介绍篇

    一 基本概念 1.lucene介绍 (1)简介 lucene是最受欢迎的java开源全文搜索引擎开发工具包.lucene使用反向索引,提供了完整的查询引擎和索引引擎,部分文本分词引擎.Lucene的目 ...

  5. javaSE学习笔记01 入门篇

    javaSE学习笔记01 入门篇 java语言概述 Java背景知识 java是 美国 sun 公司 在1995年推出的一门计算机高级编程语言. java早期称为Oak(橡树),后期改名为Java. ...

  6. 谷歌大脑科学家亲解 LSTM:一个关于“遗忘”与“记忆”的故事 本文作者:奕欣 2017-01-14 09:46 导语:AI科技评论保证这是相对通俗易懂的一篇入门介绍了,看不懂的话欢迎关注「AI 科技

    谷歌大脑科学家亲解 LSTM:一个关于"遗忘"与"记忆"的故事 本文作者:奕欣 2017-01-14 09:46 导语:AI科技评论保证这是相对通俗易懂的一篇入 ...

  7. Redis入门第一篇【介绍、安装】

    tags: Redis title: Redis入门第一篇[介绍.安装] 为什么要用Redis 我对Redis的简单理解:Redis相信学JavaEE的同学都听过这个名词,它是一个缓存数据库. Red ...

  8. ps基础教程新手入门第一篇:ps界面的介绍

    欢迎来到慕恬瑶平面设计,今天给大家介绍PS基础教程新手入门第一篇: ps界面的介绍.目的让第一次接触ps的小伙伴通过ps基础教程新手入门来了解ps界面的菜单栏,工具栏已方便日后使用. 首先,打开PS ...

  9. Python 01:Pyton历史和入门介绍

    Pyton历史和入门介绍 Python是在1991年诞生的一门面向对象.解释型计算机程序设计语言.Python能做很多事情,小到简单脚本大到后端架构设计,也可以使用python来做胶水语言.学习程序设 ...

最新文章

  1. Nokia5110液晶屏完全新手学习笔记(二)
  2. 正则表达式基本语法元字符
  3. 安装sqlserver时“试图执行未经授权的操作“的错误
  4. Java线程面试题 Top 53
  5. sklearn学习 6.聚类算法K-Means
  6. 【RS】BGP14条选路原则(1)
  7. java pdf查看_Java检查PDF文件是否损坏
  8. 解线性方程 matlab,用matlab求线性方程的解
  9. 组合模式-系统菜单的设计
  10. 深度长文探讨Join运算的简化和提速
  11. 圆角矩形不是圆:圆角的画法和二阶连续性
  12. 红外万能遥控器3.0,给家里带遥控器的家电赋能
  13. 真无线蓝牙耳机选购小技巧!2020五款优秀低延迟蓝牙耳机推荐
  14. TCP/IP第三章笔记IP网际协议
  15. 战队口号霸气押韵8字_枪战游戏战队名字大全
  16. 测一测自己的Sql能力之MYSQL的GROUPBY你弄懂了吗?
  17. VMware CentOS6.5 安装VMware Tools
  18. HyperMesh Notes
  19. Pikachu系列——RCE
  20. QML 播放 http 协议开头的视频流的一些问题DirectShowPlayerService::doPlay: Unresolved error code 8007000e

热门文章

  1. 中秋三天假期,深圳南澳西冲(西涌)海滩游
  2. 原理 数据溯源_什么是数据溯源?
  3. 啊?我这手速也太差了吧?——C++Easyx“挑战六秒”小游戏
  4. 数据库的优化以及如何提高数据库性能
  5. mac搭建网站服务器,Mac上搭建Web服务器--Apache
  6. 浏览器访问百度的整个过程
  7. 高级Android研发面试必问:Android屏幕适配全方位解析
  8. 安装noilinux说明
  9. Java实现乐观锁和悲观锁
  10. cocoscreator练手 入门 Flappy Bird 像素鸟项目(2)加入水管