当前位置: 首页 > news >正文

设计模式总览

一、设计模式

介绍

种一棵树最好的时间是十年前,其次是现在 《援助的死亡》-- 比萨·莫约
The best time to plant a tree was 10 years ago。 The second best time is now。 《dead aid》-- Dambisa Moyo 

1、创建型模式

1.1、单例模式

        确保一个类最多只有一个实例,并提供一个全局访问点,可以分为预加载和懒加载。

1.2、原型模式

        通过复制现有的实例来创建新的实例,无需知道相应类的信息。其实就相当于,在使用的时候刚好有这么一个对象,但是不能直接用,clone一个一模一样的对象出来,基本上就是原型模式。

        深拷贝:将一个对象复制以后,基本数据类型会被重新创建,而引用类型,指向的还是原来对象指向的应用。

        浅拷贝:将一个对象复制后,不论是基本数据类型还有引用类型,都是重新创建的。简单来说,就是深拷贝进行了完全彻底的复制,而浅拷贝不彻底。Java中clone,如果是基本类型则属于深拷贝,如果包含了引用类型则属于浅拷贝。

1.3、建造者模式

        将一个复杂对象的构造与它的表示分离,使得同样的创建可以有不同的表示。

1.4、简单工厂模式

1.5、工厂方法模式

1.6、抽象工厂模式

2、行为型模式

2.1、策略模式

避免出现if-else、switch;

2.2、责任链模式

俗称踢皮球。 把某一个请求放到某一个链式请求当中。请求的是会把链中的每一个对象都执行一遍。 审批流程。可以将请求和处理进行解耦,请求处理者也就是节点对象只需要关系自己内部处理的逻辑即可, 不感兴趣的内容可以继续转给下一个节点对象来处理。链式结构比较灵活,可以动态的添加或者删除节点,易于拓展,符合开闭原则。 使用的是要注意出现死循环的情况出现,链太长的化可能影响程序执行的效率。

2.3、命令模式

在项目开发的时候不会很常见。

2.4、观察者模式

观察者模式又称为发布订阅模式。

2.5、模板方法模式

定义算法的骨架,一次性实现算法中不变的部分,并将可变的部分交给子类来实现,各个子类中公共的行为被提取出来并集中到一个公共的父类中,从而避免了代码的重复。 模板方法中有一部分能力是开放给子类的,整体的方法步骤父类已经制定好了。模板方法一般通过继承实现,如果父类需要改的化,子类也需要改一遍,父类的稳定很重要。

2.6、委派模式

委托书、老板和员工、老板下达任务给员工,员工完成以后,将结果给老板。 双亲委派模型、BeanDefinition、DispatcherServlet。 委派模式属于行为性模式,代理模式属于结构型模式。 委派模式注重的结果,属于一种特殊的静态代理,全权代理。 委派模式注重的任务派遣,注重结果。 代理模式注重的是代码的增强,注重过程。

2.7、迭代器模式

迭代器模式又称为游标模式,提供一种顺序访问元素或者集合中元素的方法,而无需暴露集合内部表示,抽离集合对象的迭代行为到迭代器中,提供一致访问接口。 访问一个集合对象的内容而无需暴露它的内部表示,为遍历不通的集合结构提供一个统一的访问接口。 将集合中的迭代功能,抽离出去,集合不用关心具体的迭代方法。 简单结构不用迭代,复杂结构使用迭代器 Map、List。日常开发中用的比较少。

3、结构型模式

3.1、代理模式

代理模式分为静态代理、动态代理(JDK动态代理、Cglib动态代理)。 两者的区别:Cglib采用的是继承的方式,采用覆盖父类的方法;JDK采用的是实现一个接口。 两者的目的是一样的:都是通过生成字节码,重组成新的类。 JDK对用户来说更加复杂,目标类必须实现一个接口;Cglib对目标类没有任何要求。效率更高,底层没用用到反射。 JDK生成逻辑较为简单、执行效率低。Cglib有个坑,即为目标代理类不能有final修饰的方法,是不能被代理的。 SpringAop(面向切面编程)则使用的是动态代理。一个切面就相当于一个动态代理。

3.2、门面模式

简单来说门面模式就是代理模式,只不过是静态代理。属于一种特殊的静态代理。门面模式的重点是在于封装,静态代理的重点在于增强,不增强的静态代理即为门面模式。 门面模式和单例模式,很多时候会把门面模式做成单例模式。 门面模式的优点:简化了调用的过程、建设系统的依赖、更好的划分访问的层次、符合迪米特法则:最少知道原则。 门面模式不符合开闭原则,因为在封装内部逻辑发生变化的时候,门面需要改变。 门面模式可能会破坏单一职责原则,即所谓的万能类,会增加后期的维护成本。

3.3、装饰器模式

比继承更加具有扩展性,更加灵活,把扩展放到运行的时候,可以透明的并且动态的扩展一个类功能。可以方便的时候在运行的时候进行扩展、撤销。 装饰器模式像一种特殊的静态代理,装饰模式强调的是自身功能的拓展,是可以动态定制的拓展,用户自己说了算,是一种透明的拓展。 代理模式强掉的是增强,对用户不透明,用户没有多少自主权。 装饰器模式是继承的有利补充,装饰器模式完全遵守开闭原则,装饰器模式的代码较多,会增加系统的复杂性,动态装饰的时候,代码也更加复杂了。

3.4、享元模式

资源可以重复利用,避免系统中大量相似的对象创建,避免反复创建对象。享元模式大部分情况下都和工厂模式配合使用。共享数据库链接。 享元模式是一种池化技术,把经常使用的对象提前缓存起来。JDBC链接池内部采用享元模式,减少创建链接时候的耗时。 将现用的资源重复利用起来,轻量级的模式,是一种线程池的模式。内部状态:不变的状态,外部状态:变化的状态 享元模式经常用到系统底层的开发,以便解决系统的性能问题;系统中有大量的相似对象、需要缓冲池的场景。 内部状态:不变的内容 外部状态:变化的内容

3.5、组合模式

组合和聚合不同,组合中每个部件的生命周期是一样的,同生共死,聚合则不然,聚合可以随意拆解。组合模式也称为整体部分模式。 尽量使用组合不要用继承。心在一起叫团队,人在一起叫团伙。

3.6、适配器模式

适配器模式也叫变压器模式,适配器模式用来解决兼容的情况。可能会增加系统的复杂性。可能会降低代码的可读性。适配器模式分为类适配器、对象适配器、接口适配器, 其中多个接口可以用一个适配器实现,也可以这多个接口拆分成多个适配器单独处理,但是这样可能就增加了代码的复杂度了。

3.7、桥接模式

桥接模式是多重继承的一种替代,为了将抽象的部分和具体的部分结合起来,但是不通过继承的方式来实现。通过组合的模式来代替继承完成。 符合合成复用原则、符合开放封闭原则,但是增加了理解的难度。属于一种特殊的组合模式。

4、新设计模式

5、设计模式六大规则

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

开闭原则:Open Close Principle (OCP)

里是替换原则:Liskov Substitution Principle (LSP)

迪米特法则(最少知道原则): The Least Knowledge Principle (LKP)

接口隔离原则:Interface Segregation Principle (ISP)

依赖倒置原则:Dependency Inversion Principle(DIP)

6、设计模式总览

设计模式用的好就是架构师,用的不好就是码农

7、设计原则

每个人都有义乌捍卫、遵守和完善原则,原则可以修正,但是不能肆意妄为。

SRP: 每个软件模块的职责要单一,衡量标准是模块是否只有个被修改的原因,职责越单一,被修改的原因就越少,模块的内聚性越高,被复用可能性就越大,也更容易被理解。

OCP: 软件实体应该对扩展开放,对修改关闭。

LSP: 程序中父类型都应该可以正确地被子类型替换,里是替换原则由2008年图灵奖得主,美国第一位计算机科学女博士Barbara Liskov教授和卡内基.梅隆大学Jeannette Wing教授于1994年提出的。

ISP: 多个特定客户端接口要好于一个宽泛用途的接口。接口隔离原则认为不能强迫用户去依赖那些他们不适用的接口, 换句话说,使用多个接口比使用单一的总接口要好。

DIP: 模块之间的交互应该依赖抽象,而非实现,DIP要求高层模块不应该依赖于底层模块,二者都应该依赖于抽象,抽象不应该依赖于细节,细节应依赖于抽象。

DRY: DRY原则特指在程序设计和计算中避免重复代码。

YAGNI: 极限编程,提倡的原则,是指你自以为有用的功能,实际上都是用不到的,因此处理核心功能之外,其他的功能一概不要提前设计,这样可以大大加快开发进程。

Rule of Three:  某个功能第三次出现的时候,就有必要进行抽象化了,第一次用到某个功能时,写一个特定的解决方法,第二次用到的时候,复制上一次的代码,第三次出现的时候,才着手抽象化,写出通用的解决方法。

KISS: 真正的简单绝不是毫无设计感,上来就写代码,而是宝剑锋从磨砺出,亮剑的时候犹如一到华丽的闪电,背后却又大量的艰辛和积累,正真的简单不是不思考,而是先发散,再收敛,在纷繁复杂中,把握问题的核心。

POLA: 写代码不是写侦探小说,要的简单易懂,而不是是不是冒出个 Surprise。

相关文章:

  • 英伟达狂卖50万台GPU!AI爆火背后,是显卡的争夺
  • DeDeCMS v5.7 SP2 正式版 前台任意用户密码修改(漏洞复现)
  • pytorch 代码块积累
  • pycharm 创建vue并实现简易路由功能
  • LabVIEWL实现鸟巢等大型结构健康监测
  • Java零基础-switch条件语句
  • 算法设计与实现--贪心篇
  • 笔记-基于CH579M模块通过网线直连电脑进行数据收发(无需网络)
  • 什么是Daily Scrum?
  • 1038. 从二叉搜索树到更大和树 --力扣 --JAVA
  • Java内存缓存神器:Caffeine(咖啡因)
  • 松下、书客、明基护眼台灯值不值得买?热门护眼台灯真实测评!
  • Linux shell编程学习笔记32:declare 命令
  • MacOS + Android Studio 通过 USB 数据线真机调试
  • 【4】基于多设计模式下的同步异步日志系统-框架设计
  • 【Under-the-hood-ReactJS-Part0】React源码解读
  • angular学习第一篇-----环境搭建
  • CSS3 变换
  • exports和module.exports
  • JAVA 学习IO流
  • MySQL常见的两种存储引擎:MyISAM与InnoDB的爱恨情仇
  • Phpstorm怎样批量删除空行?
  • quasar-framework cnodejs社区
  • vue 配置sass、scss全局变量
  • vue数据传递--我有特殊的实现技巧
  • 半理解系列--Promise的进化史
  • 关于extract.autodesk.io的一些说明
  • 基于 Babel 的 npm 包最小化设置
  • 前端临床手札——文件上传
  • 实习面试笔记
  • 数组的操作
  • 我从编程教室毕业
  • 硬币翻转问题,区间操作
  • 运行时添加log4j2的appender
  • "无招胜有招"nbsp;史上最全的互…
  • #pragma once
  • (11)MSP430F5529 定时器B
  • (173)FPGA约束:单周期时序分析或默认时序分析
  • (zt)基于Facebook和Flash平台的应用架构解析
  • (附源码)ssm高校运动会管理系统 毕业设计 020419
  • (论文阅读30/100)Convolutional Pose Machines
  • (转)AS3正则:元子符,元序列,标志,数量表达符
  • *++p:p先自+,然后*p,最终为3 ++*p:先*p,即arr[0]=1,然后再++,最终为2 *p++:值为arr[0],即1,该语句执行完毕后,p指向arr[1]
  • .Net 4.0并行库实用性演练
  • .net 8 发布了,试下微软最近强推的MAUI
  • .net 后台导出excel ,word
  • .NetCore 如何动态路由
  • .NET关于 跳过SSL中遇到的问题
  • .NET开源的一个小而快并且功能强大的 Windows 动态桌面软件 - DreamScene2
  • @Controller和@RestController的区别?
  • @DependsOn:解析 Spring 中的依赖关系之艺术
  • @SpringBootApplication 包含的三个注解及其含义
  • [20180129]bash显示path环境变量.txt
  • [Android]如何调试Native memory crash issue
  • [AR Foundation] 人脸检测的流程