Java面试知识点(全)-设计模式二

news/2025/1/12 3:50:59/

Java面试知识点(全)
导航: https://nanxiang.blog.csdn.net/article/details/130640392
注:随时更新

13.模板模式

定义一个操作中算法的骨架,而将一些步骤延迟到子类中,模板方法使得子类可以不改变算法的结构即可重定义该算法的某些特定步骤。
  通俗点的理解就是 :完成一件事情,有固定的数个步骤,但是每个步骤根据对象的不同,而实现细节不同;就可以在父类中定义一个完成该事情的总方法,按照完成事件需要的步骤去调用其每个步骤的实现方法。每个步骤的具体实现,由子类完成。
Spring (FileSystemXmlApplicationContext),类路径加载配置文件(ClassPathXmlApplicationContext),以及根据项目上下文目录(XmlWebApplicationContext)加载配置文件
抽象父类(AbstractClass):实现了模板方法,定义了算法的骨架。
  具体类(ConcreteClass):实现抽象类中的抽象方法,即不同的对象的具体实现细节。
在这里插入图片描述

来举个例子: 比如我们做菜可以分为三个步骤 (1)备料 (2)具体做菜 (3)盛菜端给客人享用,这三部就是算法的骨架 ;然而做不同菜需要的料,做的方法,以及如何盛装给客人享用都是不同的这个就是不同的实现细节。
下来我们就代码实现如下
 a. 先来写一个抽象的做菜父类:

public abstract class DodishTemplate {    /*** 具体的整个过程*/protected void dodish(){this.preparation();this.doing();this.carriedDishes();}/*** 备料*/public abstract void preparation();/*** 做菜*/public abstract void doing();/*** 上菜*/public abstract void carriedDishes ();
}

b. 下来做两个番茄炒蛋(EggsWithTomato)和红烧肉(Bouilli)实现父类中的抽象方法

/*** 西红柿炒蛋* @author aries*/
public class EggsWithTomato extends DodishTemplate{@Overridepublic void preparation() {System.out.println("洗并切西红柿,打鸡蛋。");}@Overridepublic void doing() {System.out.println("鸡蛋倒入锅里,然后倒入西红柿一起炒。");}@Overridepublic void carriedDishes() {System.out.println("将炒好的西红寺鸡蛋装入碟子里,端给客人吃。");}
}/*** 红烧肉* @author aries
*/
public class Bouilli extends DodishTemplate{@Overridepublic void preparation() {System.out.println("切猪肉和土豆。");}@Overridepublic void doing() {System.out.println("将切好的猪肉倒入锅中炒一会然后倒入土豆连炒带炖。");}@Overridepublic void carriedDishes() {System.out.println("将做好的红烧肉盛进碗里端给客人吃。");}
}

c. 在测试类中我们来做菜

public class App {public static void main(String[] args) {DodishTemplate eggsWithTomato = new EggsWithTomato();eggsWithTomato.dodish();System.out.println("-----------------------------");DodishTemplate bouilli = new Bouilli();bouilli.dodish();}
}
(1)具体细节步骤实现定义在子类中,子类定义详细处理算法是不会改变算法整体结构。
(2)代码复用的基本技术,在数据库设计中尤为重要。
(3)存在一种反向的控制结构,通过一个父类调用其子类的操作,通过子类对父类进行扩展增加新的行为,符合“开闭原则”。

14、策略模式(strategy)

策略模式定义了一系列算法,并将每个算法封装起来,使他们可以相互替换,且算法的变化不会影响到使用算法的客户。需要设计一个接口,为一系列实现类提供统一的方法,多个实现类实现该接口,设计一个抽象类(可有可无,属于辅助类),提供辅助函数,关系图如下:

在这里插入图片描述

图中ICalculator提供同意的方法,

//AbstractCalculator是辅助类,提供辅助方法,接下来,依次实现下每个类:
//首先统一接口:public interface ICalculator {  public int calculate(String exp);  
}  
//辅助类:
public abstract class AbstractCalculator {  public int[] split(String exp,String opt){  String array[] = exp.split(opt);  int arrayInt[] = new int[2];  arrayInt[0] = Integer.parseInt(array[0]);  arrayInt[1] = Integer.parseInt(array[1]);  return arrayInt;  }  
}  
//三个实现类:
public class Plus extends AbstractCalculator implements ICalculator {  @Override  public int calculate(String exp) {  int arrayInt[] = split(exp,"\\+");  return arrayInt[0]+arrayInt[1];  }  
}  
public class Minus extends AbstractCalculator implements ICalculator {  @Override  public int calculate(String exp) {  int arrayInt[] = split(exp,"-");  return arrayInt[0]-arrayInt[1];  }  }  
public class Multiply extends AbstractCalculator implements ICalculator {  @Override  public int calculate(String exp) {  int arrayInt[] = split(exp,"\\*");  return arrayInt[0]*arrayInt[1];  }  
}  
//简单的测试类:
public class StrategyTest {  public static void main(String[] args) {  String exp = "2+8";  ICalculator cal = new Plus();  int result = cal.calculate(exp);  System.out.println(result);  }  
}  
输出:10

策略模式的决定权在用户,系统本身提供不同算法的实现,新增或者删除算法,对各种算法做封装。因此,策略模式多用在算法决策系统中,外部用户只需要决定用哪个算法即可。

15、模板方法模式(Template Method)

解释一下模板方法模式,就是指:一个抽象类中,有一个主方法,再定义1…n个方法,可以是抽象的,也可以是实际的方法,定义一个类,继承该抽象类,重写抽象方法,通过调用抽象类,实现对子类的调用,先看个关系图:

在这里插入图片描述

就是在AbstractCalculator类中定义一个主方法calculate,calculate()调用spilt()等,Plus和Minus分别继承AbstractCalculator类,通过对AbstractCalculator的调用实现对子类的调用,看下面的例子:

public abstract class AbstractCalculator {  /*主方法,实现对本类其它方法的调用*/  public final int calculate(String exp,String opt){  int array[] = split(exp,opt);  return calculate(array[0],array[1]);  }  /*被子类重写的方法*/  abstract public int calculate(int num1,int num2);  public int[] split(String exp,String opt){  String array[] = exp.split(opt);  int arrayInt[] = new int[2];  arrayInt[0] = Integer.parseInt(array[0]);  arrayInt[1] = Integer.parseInt(array[1]);  return arrayInt;  }  
}  
public class Plus extends AbstractCalculator {  @Override  public int calculate(int num1,int num2) {  return num1 + num2;  }  
}  
测试类:
public class StrategyTest {  public static void main(String[] args) {  String exp = "8+8";  AbstractCalculator cal = new Plus();  int result = cal.calculate(exp, "\\+");  System.out.println(result);  }  
}  

我跟踪下这个小程序的执行过程:首先将exp和"\+"做参数,调用AbstractCalculator类里的calculate(String,String)方法,在calculate(String,String)里调用同类的split(),之后再调用calculate(int ,int)方法,从这个方法进入到子类中,执行完return num1 + num2后,将值返回到AbstractCalculator类,赋给result,打印出来。正好验证了我们开头的思路。

16、观察者模式(Observer)

包括这个模式在内的接下来的四个模式,都是类和类之间的关系,不涉及到继承,学的时候应该 记得归纳,记得本文最开始的那个图。观察者模式很好理解,类似于邮件订阅和RSS订阅,当我们浏览一些博客或wiki时,经常会看到RSS图标,就这的意思是,当你订阅了该文章,如果后续有更新,会及时通知你。其实,简单来讲就一句话:当一个对象变化时,其它依赖该对象的对象都会收到通知,并且随着变化!对象之间是一种一对多的关系。先来看看关系图:

在这里插入图片描述

我解释下这些类的作用:MySubject类就是我们的主对象,Observer1和Observer2是依赖于MySubject的对象,当MySubject变化时,Observer1和Observer2必然变化。AbstractSubject类中定义着需要监控的对象列表,可以对其进行修改:增加或删除被监控对象,且当MySubject变化时,负责通知在列表内存在的对象。我们看实现代码:

//一个Observer接口:
public interface Observer {  public void update();  
}  
//两个实现类:
public class Observer1 implements Observer {  @Override  public void update() {  System.out.println("observer1 has received!");  }  
}  
public class Observer2 implements Observer {  @Override  public void update() {  System.out.println("observer2 has received!");  }  
}  
//Subject接口及实现类:
public interface Subject {  /*增加观察者*/  public void add(Observer observer);  /*删除观察者*/  public void del(Observer observer);  /*通知所有的观察者*/  public void notifyObservers();  /*自身的操作*/  public void operation();  
}  public abstract class AbstractSubject implements Subject {  private Vector<Observer> vector = new Vector<Observer>();  @Override  public void add(Observer observer) {  vector.add(observer);  }  @Override  public void del(Observer observer) {  vector.remove(observer);  }  @Override  public void notifyObservers() {  Enumeration<Observer> enumo = vector.elements();  while(enumo.hasMoreElements()){  enumo.nextElement().update();  }  }  
} 
public class MySubject extends AbstractSubject {  @Override  public void operation() {  System.out.println("update self!");  notifyObservers();  }  }  测试类:
public class ObserverTest {  public static void main(String[] args) {  Subject sub = new MySubject();  sub.add(new Observer1());  sub.add(new Observer2());  sub.operation();  }  
}  
输出:
update self!
observer1 has received!
observer2 has received!

这些东西,其实不难,只是有些抽象,不太容易整体理解,建议读者:根据关系图,新建项目,自己写代码(或者参考我的代码),按照总体思路走一遍,这样才能体会它的思想,理解起来容易!

17、迭代器模式(Iterator)

顾名思义,迭代器模式就是顺序访问聚集中的对象,一般来说,集合中非常常见,如果对集合类比较熟悉的话,理解本模式会十分轻松。这句话包含两层意思:一是需要遍历的对象,即聚集对象,二是迭代器对象,用于对聚集对象进行遍历访问。我们看下关系图:

在这里插入图片描述

这个思路和我们常用的一模一样,MyCollection中定义了集合的一些操作,MyIterator中定义了一系列迭代操作,且持有Collection实例,我们来看看实现代码:

//两个接口:
public interface Collection {  public Iterator iterator();  /*取得集合元素*/  public Object get(int i);  /*取得集合大小*/  public int size();  
}  
public interface Iterator {  //前移  public Object previous();  //后移  public Object next();  public boolean hasNext();  //取得第一个元素  public Object first();  
}  
//两个实现:
public class MyCollection implements Collection {  public String string[] = {"A","B","C","D","E"};  @Override  public Iterator iterator() {  return new MyIterator(this);  }  @Override  public Object get(int i) {  return string[i];  }  @Override  public int size() {  return string.length;  }  
}  
public class MyIterator implements Iterator {  private Collection collection;  private int pos = -1;  public MyIterator(Collection collection){  this.collection = collection;  }  @Override  public Object previous() {  if(pos > 0){  pos--;  }  return collection.get(pos);  }  @Override  public Object next() {  if(pos<collection.size()-1){  pos++;  }  return collection.get(pos);  }  @Override  public boolean hasNext() {  if(pos<collection.size()-1){  return true;  }else{  return false;  }  }  @Override  public Object first() {  pos = 0;  return collection.get(pos);  }  }  //测试类:
public class Test {  public static void main(String[] args) {  Collection collection = new MyCollection();  Iterator it = collection.iterator();  while(it.hasNext()){  System.out.println(it.next());  }  }  
}  
输出:A B C D E

此处我们貌似模拟了一个集合类的过程,感觉是不是很爽?其实JDK中各个类也都是这些基本的东西,加一些设计模式,再加一些优化放到一起的,只要我们把这些东西学会了,掌握好了,我们也可以写出自己的集合类,甚至框架!

外传

😜 原创不易,如若本文能够帮助到您的同学
🎉 支持我:关注我+点赞👍+收藏⭐️
📝 留言:探讨问题,看到立马回复
💬 格言:己所不欲勿施于人 扬帆起航、游历人生、永不言弃!🔥

http://www.ppmy.cn/news/67757.html

相关文章

ASEMI代理LT8471IFE#PBF原装ADI车规级LT8471IFE#PBF

编辑&#xff1a;ll ASEMI代理LT8471IFE#PBF原装ADI车规级LT8471IFE#PBF 型号&#xff1a;LT8471IFE#PBF 品牌&#xff1a;ADI/亚德诺 封装&#xff1a;TSSOP-20 批号&#xff1a;2023 引脚数量&#xff1a;20 工作温度&#xff1a;-40C~125C 安装类型&#xff1a;表面…

np读取txt、csv文件的数据

目录 1、基础参数 2、参数详解 3、应用参数示例 机器学习中使用np.loadtxt()可以高效的导入数据&#xff0c;np.loadtxt()适合.txt文件和.csv文件。但是它默认读取float类型的值。 1、基础参数 numpy.loadtxt(fname, dtype, comments#, delimiterNone, convertersNone, s…

NC与单一窗口数据对接丨外贸软件

在国际贸易通关过程中&#xff0c;所涉及相关部门的信息管理&#xff0c;主要是以数字化流程系统为主&#xff0c;让每个部门业务的申请、办理、回复采用电子化和互联网化。由于每个环节部分的数据壁垒未打通&#xff0c;数据无法协同共享&#xff0c;导致在口岸通关的过程中&a…

sort ,uniq ,tr ,cut命令

目录 一、sort命令 1.格式 2.命令演示 2.1 选项 -n :按照数字大小排序 2.2选项 -r &#xff1a;反向排序 2.3选项 -u&#xff1a;隐藏多余行 2.4选项 -t &#xff1a;指定分隔符&#xff1b;-k &#xff1a;指定字符&#xff1b;-o &#xff1a;指定保存文件 二、uniq命令…

详细介绍SVG.js中textPath和text元素相关的一些方法,帮助大家更好的了解和使用这两个元素

SVG.js是一个基于SVG的JavaScript库&#xff0c;可以帮助更加简单地操作SVG元素。其中&#xff0c;textPath和text元素是SVG.js中非常重要的两个元素&#xff0c;它们可以让我们在SVG图形中添加文字&#xff0c;并实现一些炫酷的效果。在本篇文章中&#xff0c;我们将详细介绍S…

计网U+选择题知识点

第一章U 1.Internet 网络本质上属于&#xff08; 分组交换&#xff09;网络。 2.(网络层发送的是分组) 3.在 OSI 参考模型中,自下而上第一个提供端到端服务的是(传输层 )。 4.APRANET 5.三类大家很熟悉的网络是&#xff08;电信网络&#xff0c;有线电视网络&#xff0c;计算机…

瑞吉外卖 - 完善后台系统登陆功能(5)

某马瑞吉外卖单体架构项目完整开发文档&#xff0c;基于 Spring Boot 2.7.11 JDK 11。预计 5 月 20 日前更新完成&#xff0c;有需要的胖友记得一键三连&#xff0c;关注主页 “瑞吉外卖” 专栏获取最新文章。 相关资料&#xff1a;https://pan.baidu.com/s/1rO1Vytcp67mcw-PD…

最新DNDC模型在土地利用变化、未来气候变化下的建模方法及温室气体时空动态模拟实践技术应用

由于全球变暖、大气中温室气体浓度逐年增加等问题的出现&#xff0c;“双碳”行动特别是碳中和已经在世界范围形成广泛影响。。DNDC&#xff08;Denitrification-Decomposition&#xff0c;反硝化-分解模型&#xff09;是目前国际上最为成功的模拟生物地球化学循环的模型之一&a…