java Spring中使用到的设计模式

单例模式

单例模式(Singleton Pattern)是java中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一对象的方式,可以直接访问,不需要实例化该类的对象。
注意:
(1)单例类只能有一个实例。
(2)单例类必须自己创建自己的唯一实例。
(3)单例类必须给所有其他对象提供这一实例。

优点:
(1)在内存里面只有一个实例,减少了内存的开销,尤其是平凡创建和销毁实例。
(2)避免对文件的多重占用(比如写文件操作)。

缺点:没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。

使用场景:
(1)要求生产唯一序列号。
(2)Web中的计数器,不用每次刷新都在数据库里加一次,用单例先缓存起来。
(3)创建的一个对象需要消耗的资源过多,比如I/O与数据库的链接等。

注意事项:getInstance()方法中需要使用同步锁synchronized(Singleton.class)防止多线程同时进入造成instance被多次实例化。

实现:

	public class SingleObject {//创建SingleOject的一个对象private static SingleObject instance = new SingleObject();//让构造器函数为private,这样类就不会被实例化private SingleObject(){};//获取唯一可用的对象public static SingleObject getInstance(){return instance;}public void showMessage(){System.out.println("aaaa");}
}

Spring中定义的bean默认为单例模式,保证每一个类中仅有一个实例,并提供一个访问它的全局访问点。spring中的单例模式提供了全局访问点BeanFactory。但没有从构造器级别去控制单例,这是因为spring管理的是任意的java对象。

单例模式的几种实现方式

(1)懒汉式,线程不安全
这种方式是最基本的实现方式,这种实现的最大的问题就是不支持多线程。因为没有加锁synchronized,所以严格意义上不算单例模式。

public class Singleton {  private static Singleton instance;  private Singleton (){}  public static Singleton getInstance() {  if (instance == null) {  instance = new Singleton();  }  return instance;  }  
}

(2)懒汉式,线程安全
能够在多线程中很好的工作,但是效率很低,99%的情况下不需要同步。
优点:第一次调用才初始化,避免内存浪费。
缺点:必须加synchronized才能保证单例,但加锁会影响效率。

public class Singleton {  private static Singleton instance;  private Singleton (){}  public static synchronized Singleton getInstance() {  if (instance == null) {  instance = new Singleton();  }  return instance;  }  
}

(3)饿汉式
这种方法比较常用,但容易产生垃圾对象。
优点:没有加锁,执行效率会提高。
缺点:类加载时就初始化,浪费内存。
它基于classloader机制避免了多线程的同步问题,不过,instance在类装载时就实例化。

public class Singleton {  private static Singleton instance = new Singleton();  private Singleton (){}  public static Singleton getInstance() {  return instance;  }  
}

(4)双检索/双重校验锁(DCL,即double-checked locking)
这种方式采用双锁机制,安全且在多线程情况下保持高性能。

public class Singleton {  private volatile static Singleton singleton;  private Singleton (){}  public static Singleton getSingleton() {  if (singleton == null) {  synchronized (Singleton.class) {  if (singleton == null) {  singleton = new Singleton();  }  }  }  return singleton;  }  
}

(5)登记式/静态内部类
这种方式能达到双检锁方式一样的功效,但实现更简单。对静态域使用延迟初始化,应该使用这种方式而不是双检锁方式。
这种方式同样使用了classLoader机制来保证初始化instace时只有一个线程,它跟第三种方式不同的是:第3种方式只要Singleton类被装载了,那么instance就会被实例化,而这种方式是Singleton类被撞在了,instance不一定被初始化。因为SingletonHolder类没有被主动使用,只有通过显式调用getInstance方法时,才会显式装载Singleton类,从而实例化instance。

public class Singleton {private static class SingletonHolder {private static finale Singleton INSTANCE = new Singleton(); }private Singleton (){};public static final Singleton getInstance(){return SingletonHolder.INSTANCE;}
}

(6)枚举
这是实现单例模式的最佳方法。它更简洁,自动支持序列化机制,防止被多次实例化。

public enum Singleton {  INSTANCE;  public void whateverMethod() {  }  
}

总结:一般情况下,不建议使用第一种和第二种懒汉方式,建议使用第三种饿汉方式,只有在明确要实现lazy loading 效果时,才会使用第五种登记方式。如果涉及到反序列化创建对象时,可以尝试使用第六种枚举方式。

工厂模式

工厂模式(Factory Pattern)是Java中最常用的设计模式之一。这种类型的设计模式属于创建型模式。它提供了一种创建对象的最佳模式。在工厂模式中,我们在创建对象时不会对客户端暴露创建逻辑,并且通过使用一个共同的接口来指向新创建的对象。

优点:
(1)一个调用者想创建一个对象,只要知道其名称就可以了。
(2)扩展性高,如果想要增加一个产品,只要扩展一个工厂类就可以了。
(3)屏蔽产品的具体实现,调用者只关心产品的接口。

注意事项:作为一种创建类模式,在任何需要生成复杂对象的地方,都可以使用工厂方法模式。有一点需要注意的地方就是复杂对象适合使用工厂模式,而简单对象,特别是只需要通过new就可以完成创建的对象,无需使用工厂模式。如果使用工程模式,就需要引入一个工厂类,会增加系统的复杂度。

使用场景:
(1)日志记录器:记录可以记录到本地磁盘、系统事件、远程服务器等,用户可以选择日志记录到什么地方。
(2)数据库访问,当用户不知道最后系统采用哪一类数据库时,以及数据库可能有变化时。
(3)设计一个连接服务器的框架,需要三个协议,“POP3”,“IMAP”,“HTTP”,可以把三个作为产品类,实现同一个接口。

Spring中使用工厂模式通过BeanFactory、ApplicationContext创建Bean对象。

实现方式:
我们创建一个Shape接口和实现Shape接口的实体类。下一步是定义工厂类ShapeFactory。
FactoryPatternDemo类使用ShapeFactory来获取Shape对象。它将向ShapeFactory传递信息(CIRCLE / RECTANGLE / SQUARE),以便获取它所需对象的类型。

步骤一:创建一个接口

public insterface Shape{void draw();
}

步骤二:创建实现接口的实体类

public class Rectangle implements Shape{@Overridepublic void draw() {System.out.println("Inside Rectangle::draw() method.");}
}
public class Square implements Shape {@Overridepublic void draw() {System.out.println("Inside Square::draw() method.");}
}
public class Circle implements Shape {@Overridepublic void draw() {System.out.println("Inside Circle::draw() method.");}
}

步骤三:创建一个工厂,生成基于给定信息的实体类的对象

public class ShapeFactory{//使用getShape方法获取形状类型的对象public Shape getShape(String shapeType) {if (shapeType == null) {return null;}if(shapeType.equalsIgnoreCase("CIRCLE")){return new Circle();} else if(shapeType.equalsIgnoreCase("RECTANGLE")){return new Rectangle();} else if(shapeType.equalsIgnoreCase("SQUARE")){return new Square();}return null;}
}

步骤四:使用该工厂,通过传递

public class FactoryPatternDemo {public static void main(String[] args) {ShapeFactory shapeFactory = new ShapeFactory();//获取 Circle 的对象,并调用它的 draw 方法Shape shape1 = shapeFactory.getShape("CIRCLE");//调用 Circle 的 draw 方法shape1.draw();//获取 Rectangle 的对象,并调用它的 draw 方法Shape shape2 = shapeFactory.getShape("RECTANGLE");//调用 Rectangle 的 draw 方法shape2.draw();//获取 Square 的对象,并调用它的 draw 方法Shape shape3 = shapeFactory.getShape("SQUARE");//调用 Square 的 draw 方法shape3.draw();}
}

抽象工厂模式

抽象工厂模式(Abstract Factory Pattern)是围绕一个超级工厂创建其他工厂,该超级工厂又称为其他工厂的工厂。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
在抽象工厂模式中,接口是负责创建一个相关对象的工厂,不需要显式地指定它们的类,每个生成的工厂都能按照工厂模式提供对象。

优点:当一个产品族中的多个对象被设计成一起工作时,它能保证客户端始终只使用同一个产品族中的对象。
缺点:产品族扩展非常困难,要增加一个系列的某一产品,既要在抽象的Creator里加代码,又要在具体的里面加代码。

使用场景:
(1)QQ换皮肤,一整套一起换
(2)生成不同操作系统的程序

注意事项:
产品族难扩展, 产品等级易扩展

策略模式

在策略模式中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为型模式。
在策略模式中,我们创建表示各种策略的对象和一个行为随着策略对象改变而改变的context对象。策略对象改变context对象的执行算法。

意图:定义一系列的算法,把他们一个个封装起来,并且使他们可相互替换。
主要解决:在多种算法相似的情况下,使用if…else所带来的复杂和难以维护。
何时使用:一个系统有许多类,而区分它们的只是他们直接的行为。
如何解决:将这些算法封装成一个一个的类,任意的替换。

优点:
(1)算法可以自由切换。
(2)避免使用多重条件判断。
(3)扩展性良好。

缺点:
(1)策略类会增多。
(2)所有策略类都需要对外暴露。

Spring中资源访问接口Resource的设计是一种经典的策略模式。Resource接口是所有资源访问类所实现的接口,Resource接口就代表资源访问策略,但具体采用哪种策略实现,Resource接口并不理会。客户端程序只和Resource接口耦合,并不知道底层采用何种资源访问策略,这样客户端可以再不同的资源访问策略之间自由切换。

实现方法:
创建一个定义活动的Strategy接口和实现了Strategy

步骤一:

创建一个接口:

public interface Strategy {public int doOperation(int num1, int num2);
}

步骤二

创建接口的实现类

public class OperationAdd implements Strategy{@Overridepublic int doOperation(int num1, int num2) {return num1 + num2;}
}
public class OperationSubtract implements Strategy{@Overridepublic int doOperation(int num1, int num2) {return num1 - num2;}
}
public class OperationMultiply implements Strategy{@Overridepublic int doOperation(int num1, int num2) {return num1 * num2;}
}

步骤三

创建Context类

public class Context {private Strategy strategy;public Context(Strategy strategy){this.strategy = strategy;}public int executeStrategy(int num1, int num2){return strategy.doOperation(num1, num2);}
}

步骤四

使用Context来 查看当它改变策略Strategy时的行为变化

public class StrategyPatternDemo {public static void main(String[] args) {Context context = new Context(new OperationAdd());    System.out.println("10 + 5 = " + context.executeStrategy(10, 5));context = new Context(new OperationSubtract());      System.out.println("10 - 5 = " + context.executeStrategy(10, 5));context = new Context(new OperationMultiply());    System.out.println("10 * 5 = " + context.executeStrategy(10, 5));}
}

代理模式

在代理模式中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/3333.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

电压放大器在超声波焊接中的作用以及应用

电压放大器是一种运用于电子设备中的信号放大器,主要作用是将小信号放大为更高幅度的信号。在超声波焊接中,电压放大器起到了重要的作用,它可以将从传感器采集到的微小信号放大为能够被检测和处理的合适大小的信号。 超声波焊接是现代工业生产…

微信怎么自动加好友,通过好友后自动打招呼

很多客户朋友每天花大量的时间用手机搜索添加好友,这样的添加很集中也容易频繁,而且效率还低。对方通过后,有时也不能及时和客户搭建链接,导致客户也流失了。 现在可以实现自动添加和自动打招呼哦,只需要导入数据、设置…

【从零开始学CSS | 第二篇】伪类选择器

目录 前言: 伪类选择器: 常见的伪类选择器: 举例: 小窍门: 总结: 前言: 上一篇文章我们详细的为大家介绍了一些常见的选择器,这几篇我们将再次介绍CSS中的一个常见选择器——伪类选择器&am…

设计模式之适配器模式

写在前面 适配器设计模式属于结构型设计模式的一种,本文一起来看下。 1:介绍 1.1:什么时候适配器设计模式 当现有接口客户端无法直接调用时,我们可以考虑适配器设计模式,来定义一个能够供客户端直接调用的接口&…

软件测试的分类

代码分类: 1、黑盒测试 2、白盒测试 3、灰黑测试 黑盒测试: 把测试的对象看成是一个黑色的盒子的,看不到里面内部的结构,是对软件的一种功能性的测试。 白盒测试: 就是把测试的对象看成是一个透明的盒子&#x…

测试老鸟总结,性能测试-最佳并发和最大并发,性能测试实施...

目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 性能测试&#xf…

curl操作

下载路径:https://curl.se/windows/ 参考:https://blog.csdn.net/weixin_45191386/article/details/130652821 操作: curl http://localhost:8085/api/v1/aaa/bbbb/?ccc 652781344055627776

Maven下载依赖的顺序及配置文件说明

在 Maven 中,当下载依赖项时,存在多个仓库时会按照以下优先级顺序进行搜索: 本地仓库:Maven 会首先在本地的 Maven 仓库中查找依赖项。 私有仓库(私服):如果在本地仓库中未找到依赖项&#xf…

编程导航算法村第二关 | 白银挑战

编程导航算法村第二关 | 白银挑战 指定区间的链表翻转 LeetCode92 &#xff1a;给你单链表的头指针 head 和两个整数 left 和 right &#xff0c;其中 left < right。请你反转从位置 left 到位置 right 的链表节点&#xff0c;返回反转后的链表。 public ListNode revers…

第四次CCF计算机软件能力认证

第一题&#xff1a;图像旋转 旋转是图像处理的基本操作&#xff0c;在这个问题中&#xff0c;你需要将一个图像逆时针旋转 90 度。 计算机中的图像表示可以用一个矩阵来表示&#xff0c;为了旋转一个图像&#xff0c;只需要将对应的矩阵旋转即可。 输入格式 输入的第一行包含两…

我国版式文档格式OFD前端WEB展示之EasyOFD

EasyOFD an ofd file web shower 一个在web端展示ofd文件的控件&#xff0c;该控件基于CANVAS绘制。 该控件使用了以下外部程序 1&#xff09;jszip&#xff1a;解决解压文件。 2&#xff09;x2js: 解决XML文件到JS转换 3&#xff09;easyjbig2: 解决ofd内部使用jb2文件存储的…

NSSCTF刷web(2)

[NISACTF 2022]bingdundun~ bingdundun处感觉像文件包含,改upload为index 发现确实,猜测会补一个后缀.php 那常规文件包含都不行了,这里还有一个文件上传的功能,考虑phar协议 <?php$phar new Phar("test.phar"); $phar->startBuffering(); $phar->setStu…

小马识途分享百度百科收录词条的规则

百度百科词条是人人都可以编辑的&#xff0c;并且都是免费创建&#xff0c;但是自己创建百科词条往往审核不通过&#xff0c;一般企业会把这项任务委托给有经验的营销团队。这里小马识途营销顾问分享一下百度百科收录词条的规则。 百度百科收录规则主要分为&#xff1a;规范词条…

【excel细碎小知识点】

目录索引 &符号的用法&#xff1a;实例演示&#xff1a; 数字显示和位数的区别&#xff1a;分列功能的妙用&#xff1a;什么叫做常规类型&#xff1a; &符号的用法&#xff1a; **连接字符串:**转化后都是文本字符串类型。你可以通过修改数据类型进行更多可能的操作 实…

【Go语言开发】简单了解一下搜索引擎并用go写一个demo

写在前面 这篇文章我们一起来了解一下搜索引擎的原理&#xff0c;以及用go写一个小demo来体验一下搜索引擎。 简介 搜索引擎一般简化为三个步骤 爬虫&#xff1a;爬取数据源&#xff0c;用做搜索数据支持。索引&#xff1a;根据爬虫爬取到的数据进行索引的建立。排序&#xf…

prometheus调整默认数据存储时间

调整kubernetes部署的prometheus数据存储时间 由于prometheus是用kuberentes部署的&#xff0c;没办法像传统部署方式那种直接在启动参数增加存储时间的参数。需要在configmap里或者在deployment里添加&#xff0c;我这里使用的方式是在deployement里添加调整存储时间的参数。…

学会在重装系统前如何备份软件,再也不怕失去珍贵的应用!

​Windows系统是电脑的重要组成部分&#xff0c;它不仅提供了友好的用户界面&#xff0c;还承担着许多关键的功能和任务&#xff0c;为我们提供了一个稳定、安全和效率的工作环境&#xff0c;使我们能够充分发挥电脑的潜力&#xff0c;优化工作效率和生活品质。 随着系统使…

为 GitHub 设置 SSH 密钥

1. 起因 给自己的 github 改个名&#xff0c;顺便就给原来 Hexo 对应的仓库也改了个名。然后发现 ubhexo clean && hexo generate && hexo deploy 失败了&#xff0c;报错如下&#xff1a; INFO Deploying: git INFO Clearing .deploy_git folder... INFO …

软件渗透测试真的很重要吗?渗透测试有哪些测试流程?

软件渗透测试是指通过模拟恶意攻击者的行为&#xff0c;评估软件系统中的潜在安全漏洞和弱点的活动。这种安全测试方法能够帮助开发人员和系统管理员发现并修复潜在的安全漏洞&#xff0c;以确保软件系统的安全性和完整性。软件渗透测试是一项高度技术性的任务&#xff0c;需要…

javaee jsp页面 九大内置对象和四大作用域

九大内置对象四大域 一、四大域 域对象的作用:保存数据,获取数据,共享数据 作用域从小到大为&#xff1a;PageContext&#xff08;jsp页面&#xff09;&#xff0c;ServletRequest&#xff08;一次请求&#xff09;&#xff0c;HttpSession&#xff08;一次会话&#xff09;&am…