Java面试题 - 设计模式

简介: Java面试题 - 设计模式
01 工厂方法模式(利用创建同一接口的不同实例)

普通⼯⼚模式:建⽴⼀个⼯⼚类,对实现了同⼀接⼝的⼀些类进⾏实例的创建;

多个⼯⼚⽅法模式:提供多个⼯⼚⽅法,分别创建对象;

静态⼯⼚⽅法模式:将上⾯的多个⼯⼚⽅法置为静态的,不需要创建⼯⼚实例,直接调⽤即可;

适⽤场景:凡是出现了⼤量不同种类的产品需要创建,并且具有共同的接⼝时,可以通过⼯⼚⽅法模式进⾏创建。在以上的三种模式中,第⼀种如果传⼊的字符串有误,不能正确创建对象,第三种相对于第⼆种,不需要实例化⼯⼚类,所以,⼤多数情况下,我们会选⽤第三种——静态⼯⼚⽅法模式。

02 抽象工厂模式(多个工厂)

创建多个⼯⼚类,提⾼⼯⼚的扩展性,不⽤像上⾯⼀样如果增加产品则要去修改唯⼀的⼯⼚类;

03 单例模式(保证对象只有⼀个实例)

描述:保证在⼀个JVM中,该对象只有⼀个实例存在;

适⽤场景:

  1. 某些类创建⽐较频繁,对于⼀些⼤型的对象,这是⼀笔很⼤的系统开销。
  2. 省去了new操作符,降低了系统内存的使⽤频率,减轻GC压⼒。
  3. 有些类如交易所的核⼼交易引擎,控制着交易流程,如果该类可以创建多个的话,系统完全乱了。(⽐如⼀个军队出现了多个司令员同时指挥,肯定会乱成⼀团),所以只有使⽤单例模式,才能保证核⼼交易服务器独⽴控制整个流程。

分类:

  • 饿汉式:类初始化时创建单例,线程安全,适⽤于单例占内存⼩的场景,否则推荐使⽤懒汉式延迟加载;
  • 懒汉式:需要创建单例实例的时候再创建,需要考虑线程安全(性能不太好);
  • 双重检验锁::效率⾼;(解决问题:假如两个线程A、B,A执⾏了if (instance == null)语句,它会认为单例对象没有创建,此时线程切到B也执⾏了同样的语句,B也认为单例对象没有创建,然后两个线程依次执⾏同步代码块,并分别创建了⼀个单例对象。)
  • 静态内部类⽅式:可以同时保证延迟加载和线程安全。
  • 枚举:使⽤枚举除了线程安全和防⽌反射调⽤构造器之外,还提供了⾃动序列化机制,防⽌反序列化的时候创建新的对象。
04 原型模式(对一个原型对象进行复制,克隆产生类似新对象)

描述:对⼀个原型对象进⾏复制、克隆产⽣类似新对象):将⼀个对象作为原型,对其进⾏复制、克隆,产⽣⼀个和元对象类似的新对象;

核⼼:它的核⼼是原型类Prototype,需要实现Cloneable接⼝,和重写Object类中的clone⽅法;

作⽤:使⽤原型模式创建对象⽐直接new⼀个对象在性能上要好的多,因为Object类的clone⽅法是⼀个本地⽅法,它直接操作内存中的⼆进制流,特别是复制⼤对象时,性能的差别⾮常明显。

05 适配器模式(接口兼容)

描述:将某个类的接⼝转换成客户端期望的另⼀个接⼝表示,⽬的是消除由于接⼝不匹配所造成的类的兼容性问题。

分配

  • 类的适配器模式:有一个Source类,拥有一个方法,待适配,目标接口时Targetable,通过Adapter类,将Source的功能扩展到Targettable里。
  • 对象适配器模式
  • 接口的适配器模式

使用场景:

  1. 类的适配器模式:当希望将⼀个类转换成满⾜另⼀个新接⼝的类时,可以使⽤类的适配器模式,创建⼀个新类,继承原有的类,实现新的接⼝即可。
  2. 对象的适配器模式:当希望将⼀个对象转换成满⾜另⼀个新接⼝的对象时,可以创建⼀个Wrapper类,持有原类的⼀个实例,在Wrapper类的⽅法中,调⽤实例的⽅法就⾏。
  3. 接⼝的适配器模式:当不希望实现⼀个接⼝中所有的⽅法时,可以创建⼀个抽象类Wrapper,实现所有⽅法,我们写别的类的时候,继承抽象类即可。
06 装饰模式(给对象动态增加新功能,需持有对象实例)

描述:装饰模式就是给⼀个对象增加⼀些新的功能,⽽且是动态的,要求装饰对象和被装饰对象实现同⼀个接⼝,装饰对象持有被装饰对象的实例:

使用场景:

  1. 需要扩展⼀个类的功能。
  2. 动态的为⼀个对象增加功能,⽽且还能动态撤销。(继承不能做到这⼀点,继承的功能是静态的,不能动态增删。)
07 代理模式(持有被代理类的实例,进行操作前后控制)

描述:采⽤⼀个代理类调⽤原有的⽅法,且对产⽣的结果进⾏控制。

08 外观模式(集合所有操作到一个类)

描述:外观模式是为了解决类与类之间的依赖关系的,像spring⼀样,可以将类和类之间的关系配置到配置⽂件中,⽽外观模式就是将他们的关系放在⼀个Facade类中,降低了类类之间的耦合度。

09 桥接模式(数据库驱动桥接)

桥接模式就是把事物和其具体实现分开,使他们可以各⾃独⽴的变化。桥接的⽤意是:将抽象化与实现化解耦,使得⼆者可以独⽴变化,像我们常⽤的JDBC桥DriverManager⼀样,JDBC进⾏连接数据库的时候,在各个数据库之间进⾏切换,基本不需要动太多的代码,甚⾄丝毫不⽤动,原因就是JDBC提供统⼀接⼝,每个数据库提供各⾃的实现,⽤⼀个叫做数据库驱动的程序来桥接就⾏了。

10 组合模式(部分整体模式)

组合模式有时⼜叫部分-整体模式在处理类似树形结构的问题时⽐较⽅便。

11 享元模式(共享池、数据库连接池)

享元模式的主要⽬的是实现对象的共享,即共享池,当系统中对象多的时候可以减少内存的开销,通常与⼯⼚模式⼀起使⽤。当⼀个客户端请求时,⼯⼚需要检查当前对象池中是否有符合条件的对象,如果有,就返回已经存在的对象,如果没有,则创建⼀个新对象,如数据库连接池;

12 策略模式(多种算法封装)

策略模式定义了⼀系列算法,并将每个算法封装起来,使他们可以相互替换,且算法的变化不会影响到使⽤算法的客户,需要设计⼀个接⼝,为⼀系列实现类提供统⼀的⽅法,多个实现类实现该接⼝。

13 模板方法模式(抽象方法作为骨架,具体逻辑让子类实现)

定义⼀个操作中算法的框架,⽽将⼀些步骤延迟到⼦类中,使得⼦类可以不改变算法的结构即可重定义该算法中的某些特定步骤。完成公共动作和特殊动作的分离。

14 观察者模式(发布-订阅模式)

当⼀个对象变化时,其它依赖该对象的对象都会收到通知,并且随着变化!对象之间是⼀种⼀对多的关系。类似于邮件订阅和RSS订阅,当你订阅了该⽂章,如果后续有更新,会及时通知你。

15 迭代器模式(遍历集合)

迭代器模式就是顺序访问聚集中的对象。

16 责任链模式(多任务形成⼀条链,请求在链上传递)

有多个对象,每个对象持有对下⼀个对象的引⽤,这样就会形成⼀条链,请求在这条链上传递,直到某⼀对象决定处理该请求。但是发出者并不清楚到底最终那个对象会处理该请求,所以,责任链模式可以实现,在隐瞒客户端的情况下,对系统进⾏动态的调整。

17 命令模式(实现请求和执⾏的解耦)

命令模式的⽬的就是达到命令的发出者和执⾏者之间解耦,实现请求和执⾏分开,熟悉Struts的同学应该知道,Struts其实就是⼀种将请求和呈现分离的技术,其中必然涉及命令模式的思想!

18 备忘录模式(保存和恢复对象状态)

主要⽬的是保存⼀个对象的某个状态,以便在适当的时候恢复对象。

19 状态模式(对象状态改变时改变其⾏为)

当对象的状态改变时,同时改变其⾏为。状态模式就两点:

  1. 可以通过改变状态来获得不同的⾏为。
  2. 你的好友能同时看到你的变化。
20 访问者模式(数据接⼝稳定,但算法易变)

访问者模式把数据结构和作⽤于结构上的操作解耦合,使得操作集合可相对⾃由地演化。

访问者模式适⽤于数据结构相对稳定算法⼜易变化的系统。因为访问者模式使得算法操作增加变得容易。访问者模式就是⼀种分离对象数据结构与⾏为的⽅法,通过这种分离,可达到为⼀个被访问者动态添加新的操作⽽⽆需做其它的修改的效果。

21 中介者模式

中介者模式也是⽤来降低类类之间的耦合的。如果使⽤中介者模式,只需关⼼和Mediator类的关系,具体类类之间的关系及调度交给Mediator就⾏,这有点像spring容器的作⽤。

22 解释器模式(对于⼀些固定⽂法构建⼀个解释句⼦的解释器,如正则表达式)

解释器模式⽤来做各种各样的解释器,如正则表达式等的解释器。

23 建造者模式(创建复合对象)

⼯⼚类模式提供的是创建单个类的模式,⽽建造者模式则是将各种产品集中起来进⾏管理,⽤来创建复合对象,所谓复合对象就是指某个类具有不同的属性。

24 设计模式的六大原则
  • 开闭原则(Open Close Principle):开闭原则就是说对扩展开放,对修改关闭。在程序需要进⾏拓展的时候,不能去修改原有的代码,实现⼀个热插拔的效果。所以⼀句话概括就是:为了使程序的扩展性好,易于维护和升级。想要达到这样的效果,我们需要使⽤接⼝和抽象类,后⾯的具体设计中我们会提到这点。
  • ⾥⽒代换原则(Liskov Substitution Principle):⾥⽒代换原则(Liskov Substitution Principle LSP)⾯向对象设计的基本原则之⼀。 ⾥⽒代换原则中说,任何基类可以出现的地⽅,⼦类⼀定可以出现。 LSP是继承复⽤的基⽯,只有当衍⽣类可以替换掉基类,软件单位的功能不受到影响时,基类才能真正被复⽤,⽽衍⽣类也能够在基类的基础上增加新的⾏为。⾥⽒代换原则是对“开-闭”原则的补充。实现“开-闭”原则的关键步骤就是抽象化。⽽基类与⼦类的继承关系就是抽象化的具体实现,所以⾥⽒代换原则是对实现抽象化的具体步骤的规范。
  • 依赖倒转原则(Dependence Inversion Principle):这个是开闭原则的基础,具体内容:真对接⼝编程,依赖于抽象⽽不依赖于具体。
  • 接⼝隔离原则(Interface Segregation Principle):这个原则的意思是:使⽤多个隔离的接⼝,⽐使⽤单个接⼝要好。还是⼀个降低类之间的耦合度的意思,从这⼉我们看出,其实设计模式就是⼀个软件的设计思想,从⼤型软件架构出发,为了升级和维护⽅便。所以上⽂中多次出现:降低依赖,降低耦合。
  • 迪⽶特法则(最少知道原则)(Demeter Principle):为什么叫最少知道原则,就是说:⼀个实体应当尽量少的与其他实体之间发⽣相互作⽤,使得系统功能模块相对独⽴。
  • 合成复⽤原则(Composite Reuse Principle):原则是尽量使⽤合成/聚合的⽅式,⽽不是使⽤继承。
25 JDK中的设计模式

单例模式:

java.lang.Runtime#getRuntime()
java.awt.Desktop#getDesktop()
java.lang.System#getSecurityManager()

责任链模式:

java.util.logging.Logger#log()
javax.servlet.Filter#doFilter()

观察者模式:

java.util.Observer/ java.util.Observable(很少在现实世界中使⽤)
所有实现java.util.EventListener(因此实际上各地的Swing)
javax.servlet.http.HttpSessionBindingListener
javax.servlet.http.HttpSessionAttributeListener
javax.faces.event.PhaseListener
26 Spring中的设计模式
  • 简单⼯⼚:spring中的BeanFactory就是简单⼯⼚模式的体现,根据传⼊⼀个唯⼀的标识来获得bean对象,但是否是在传⼊参数后创建还是传⼊参数前创建这个要根据具体情况来定。
  • 单例模式:Spring下默认的bean均为singleton。
  • 代理模式:为其他对象提供⼀种代理以控制对这个对象的访问。 从结构上来看和Decorator模式类似,但Proxy是控制,更像是⼀种对功能的限制,⽽Decorator是增加职责。 spring的Proxy模式在aop中有体现,⽐如JdkDynamicAopProxy和Cglib2AopProxy。
  • 观察者模式:定义对象间的⼀种⼀对多的依赖关系,当⼀个对象的状态发⽣改变时,所有依赖于它的对象都得到通知并被⾃动更新。spring中Observer模式常⽤的地⽅是listener的实现。如ApplicationListener。
目录
相关文章
|
26天前
|
Java 程序员
Java社招面试中的高频考点:Callable、Future与FutureTask详解
大家好,我是小米。本文主要讲解Java多线程编程中的三个重要概念:Callable、Future和FutureTask。它们在实际开发中帮助我们更灵活、高效地处理多线程任务,尤其适合社招面试场景。通过 Callable 可以定义有返回值且可能抛出异常的任务;Future 用于获取任务结果并提供取消和检查状态的功能;FutureTask 则结合了两者的优势,既可执行任务又可获取结果。掌握这些知识不仅能提升你的编程能力,还能让你在面试中脱颖而出。文中结合实例详细介绍了这三个概念的使用方法及其区别与联系。希望对大家有所帮助!
163 60
|
2天前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
35 14
|
5天前
|
安全 Java 程序员
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
34 13
|
25天前
|
算法 安全 Java
Java线程调度揭秘:从算法到策略,让你面试稳赢!
在社招面试中,关于线程调度和同步的相关问题常常让人感到棘手。今天,我们将深入解析Java中的线程调度算法、调度策略,探讨线程调度器、时间分片的工作原理,并带你了解常见的线程同步方法。让我们一起破解这些面试难题,提升你的Java并发编程技能!
65 16
|
22天前
|
Java 程序员 调度
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
55 9
|
27天前
|
安全 Java 程序员
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
60 12
|
1月前
|
监控 Dubbo Java
Java Dubbo 面试题
Java Dubbo相关基础面试题
|
1月前
|
SQL Java 数据库连接
Java MyBatis 面试题
Java MyBatis相关基础面试题
|
1月前
|
存储 监控 算法
Java JVM 面试题
Java JVM(虚拟机)相关基础面试题
|
1月前
|
SQL 监控 druid
Java Druid 面试题
Java Druid 连接池相关基础面试题

热门文章

最新文章