一.设计模式的思维导向图
二. 首先科普一下设计模式的七大原则:
设计模式(面向对象)有七大原则,分别是:
1.开放-封闭原则
通俗:对扩展开发,对修改关闭
2.单一职责原则
通俗:一个类只做一件事
3.依赖倒转原则
通俗:类似IOC,采用接口编程
4.迪米特法则(也称为最小知识原则)
通俗:高内聚,低耦合
5.接口隔离原则
通俗:细节接口
6.合成/聚合复用原则
通俗:避免使用继承
7.里氏代换原则
通俗:子类不能去修改父类的功能
三、结构性模式:
1、适配器模式:
常用于将一个新接口适配旧接口
PS:在我们业务代码中经常有新旧接口适配需求,可以采用该模式。
2、桥接模式:
将抽象和抽象的具体实现进行解耦,这样可以使得抽象和抽象的具体实现可以独立进行变化。
PS:这个模式,其实我们每天都在用到,但是你可能却浑然不知。只要你用到面向接口编程,其实都是在用桥接模式。
3、组合模式
让客户端看起来在处理单个对象和对象的组合是平等的,换句话说,某个类型的方法同时也接受自身类型作为参数。(So in other words methods on a type accepting the same type)
PS:从上面那句英文我们就可以得知,组合模式常用于递归操作的优化上,比如每个公司都有个boss系统,都会有什么菜单的功能。比如一级菜单下有二级菜单,二级菜单又有三级菜单。删除一级菜单的时候需要不断删除子菜单,那么这个设计模式你可以试试。总之,凡是有级联操作的,你都可以尝试这个设计模式。
4、装饰者模式
动态的给一个对象附加额外的功能,因此它也是子类化的一种替代方法。该设计模式在JDK中广泛运用,以下只是列举一小部分
PS:这个模式使用就太广了,我们常用的AOP,既有动态代理,也有装饰者的味道。
5、门面模式
为一组组件,接口,抽象或子系统提供简化的接口。
PS:我们每天使用的SLFJ日志就是门面日志,比如我们使用Dubbo,向外提供的服务就尽量采用门面模式,然后服务在调用各种service做聚合。
SLF4J,即简单日志门面(Simple Logging Facade for Java),不是具体的日志解决方案,它只服务于各种各样的日志系统。按照官方的说法,SLF4J是一个用于日志系统的简单Facade,允许最终用户在部署其应用时使用其所希望的日志System.
实际上,SLF4J所提供的核心API是一些接口以及一个LoggerFactory的工厂类。从某种程度上,SLF4J有点类似JDBC,不过比JDBC更简单,在JDBC中,你需要指定驱动程序,而在使用SLF4J的时候,不需要在代码中或配置文件中指定你打算使用那个具体的日志系统。如同使用JDBC基本不用考虑具体数据库一样,SLF4J提供了统一的记录日志的接口,只要按照其提供的方法记录即可,最终日志的格式、记录级别、输出方式等通过具体日志系统的配置来实现,因此可以在应用中灵活切换日志系统。
6、享元模式
使用缓存来减少对小对象的访问时间
PS:只要用到了缓存,基本都是在使用享元模式。很多同学都说自己的项目太low了,都没有用到什么设计模式,这不是开玩笑吗,你用个map缓存几个对象,基本上都运用了享元的思想。
7、代理模式
代理模式用于向较简单的对象代替创建复杂或耗时的对象。
PS:代理模式用得很广泛,基本所有大家知道的开源框架,都用到了动态代理。
二、创建模式
1、抽象工厂模式
抽象工厂模式提供了一个协议来生成一系列的相关或者独立的对象,而不用指定具体对象的类型。它使得应用程序能够和使用的框架的具体实现进行解耦。在JDK和许多开源框架,比如Spring中随处可见,它们很容易被发现。任何用于创建对象但返回接口或抽象类的,就是抽象工厂模式了。(any method that is used to create an object but still returns a interface or abstract class)
PS:从英文就可以得出,该模式可以与策略模式结合使用。
2、建造者模式
用于通过定义一个类来简化复杂对象的创建,该类的目的是构建另一个类的实例。构建器模式还允许实现Fluent接口。
PS:这个在我们业务代码中使用的场景太广泛了。比如订单系统大部分项目都有,订单对象就是一个复杂对象,我们就可以采用建造者模式来做。
3、工厂方法
只是一个返回实际类型的方法。
PS:这个属于大家都会的设计模式,不多过介绍。
4、原型模式
使得类的实例能够生成自身的拷贝。如果创建一个对象的实例非常复杂且耗时时,就可以使用这种模式,而不重新创建一个新的实例,你可以拷贝一个对象并直接修改它。
PS:这个你以为是冷门的设计模式,其实错了,这个是大热门的设计模式。比如我们业务代码,经常要各种DTO、BO、DO、VO转换,其实就可以参考原型设计模式的思想来做。
5、单例模式
用来确保类只有一个实例。Joshua Bloch在Effetive Java中建议到,还有一种方法就是使用枚举。
PS:在平时开发中,单例是我们用得最多的了,因为Spring的bean,默认就是单例级别的。单例属于大家基本都会的设计模式。
三、行为模式
1、责任链
通过把请求从一个对象传递到链条中下一个对象的方式来解除对象之间的耦合,直到请求被处理完毕。链中的对象是同一接口或抽象类的不同实现。
PS:凡是带有Filter
关键词的,基本都在用这个设计模式。在业务代码使用的场景实在是太多了,用到拦截器的地方基本都在用这个设计模式。
2、命令模式
将命令包装在对象中,以便可以将其存储,传递到方法中,并像任何其他对象一样返回。
PS:命令模式使用频率较高,和策略模式比较像,具体区别可以搜索一下。如果用过Activiti
工作流引擎的朋友可以看一下里面的源码,很多地方都用到了命令模式。
3、解释器模式
此模式通常描述为该语言定义语法并使用该语法来解释该格式的语句。(This pattern generally describes defining a grammar for that language and using that grammar to interpret statements in that format.)
PS:这个比较冷门,肥朝没怎么用过,你用过的话可以留言告诉肥朝。
4、迭代器模式
提供一个统一的方式来访问集合中的对象。
PS:这个中间件和基础框架组的同学可能用得比较多,业务代码的话用得不多,不过JDK中的这种使用很经典,可以看看。
5、中介者模式
通过使用一个中间对象来进行消息分发以及减少类之间的直接依赖。
PS:看到这个描述不用我多说什么,业务代码使用的场景太多了。比如你们用MQ,其实就是在用中介者模式。所以肥朝一再强调,即使是每天CRUD,关注肥朝一起学习,也能给你的CRUD项目,加上美颜+滤镜(设计模式)的加强效果。
6、备忘录模式
生成对象状态的一个快照,以便对象可以恢复原始状态而不用暴露自身的内容。比如Date对象通过自身内部的一个long值来实现备忘录模式。
PS:这个在业务中使用得不多,据肥朝了解其中一种场景是,你要把数据丢到MQ,但是MQ暂时不可用,那么你把数据暂存到DB,后面再轮询丢到MQ。如果你有更好的场景,留言告诉肥朝。
7、空对象模式
它允许您抽象空对象的处理。
PS:这个业务代码用得不多,但是JDK中的这几个方法我们倒是挺常用的。
8、观察者模式
用于为组件提供一种灵活地向感兴趣的接收者广播消息的方式。
PS:我们业务代码一般是基于Zookeeper来做观察者的。基本上用到ZK的地方,都是在用观察者模式,比如分布式锁,比如服务发现等。
9、状态模式
允许您在运行时根据内部状态轻松更改对象的行为。
PS:这个在业务代码用得就太广泛了,我就不信你们系统还没有“状态”了。比如我们常见的订单状态或者各种XX状态,都可以用得上。
10、策略模式
使用这个模式来将一组算法封装成一系列对象。通过调用这些对象可以灵活的改变程序的功能。
PS:这个太高频了,常用于优化大量的if-else
,如果这个设计模式都不会,出去不要说关注过肥朝的公众号!
11、模板方法模式
让子类可以重写方法的一部分,而不是整个重写,你可以控制子类需要重写那些操作。
PS:这个模式也是非常高频的模式。业务代码中经常遇到有很多相同的部分,我们可以做一个抽象类,子类来实现差异化,如果还不知道的,赶紧搜索一下,再次强调,非常高频。
12、访问者模式
提供一个方便的可维护的方式来操作一组对象。它使得你在不改变操作的对象前提下,可以修改或者扩展对象的行为。
PS:这个肥朝使用的频率不高,如果你有在业务代码中使用,欢迎留言告诉肥朝。
四、写在最后
你项目都用到了哪些设计模式?亦或者上述设计模式你在实际业务代码中有更好的场景,欢迎留言。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步