- 使用 Spring Initializr 创建 Spring Boot 应用程序
- 在Spring Boot中配置Cassandra
- 在 Spring Boot 上配置 Tomcat 连接池
- 将Camel消息路由到嵌入WildFly的Artemis上
GoF23 (二十三种设计模式)可按照:创建型模式、结构性模式、行为型模式三类
创建型模式:
结构性模式:
行为型模式:
初学设计模式:让我们从单例模式开始:
单例模式使⽤场景:
业务系统全局只需要⼀个对象实例,⽐如发号器、 redis 连接对象等
Spring IOC容器中的 bean 默认就是单例
spring boot 中的controller、service、dao层中通过 @Autowired的依赖注⼊对象默认都是单例的
单例模式分类:
懒汉:就是所谓的懒加载,延迟创建对象,需要用的时候再创建对象
饿汉:与懒汉相反,提前创建对象
类加载的时候直接创建对象,不管对象是否会用到!
package com.sqx.gof.single;
/***
* 饿汉式单例,类加载的时候直接创建对象,不管对象是否会用到!
*
* 存在问题:对象如果没有被使用那么,其中data1~4 会占用我们的内存!
*/
public class HungrySingle {
private byte[] data1 = new byte[1024*1024] ;
private byte[] data2 = new byte[1024*1024] ;
private byte[] data3 = new byte[1024*1024] ;
private byte[] data4 = new byte[1024*1024] ;
private HungrySingle(){ //私有构造,只能再类的内部使用,外部不可通过new创造对象
}
private static HungrySingle HUNGRY = new HungrySingle(); //类加载的时候执行
public static HungrySingle getInstance(){
return HUNGRY ;
}
}
需要获取对象的时候,才去创建对象!
package com.sqx.gof.single;
/***
* 懒汉式单例
*/
@SuppressWarnings("ALL")
public class LazySingle {
private LazySingle(){ //私有化构造
System.out.println("当前线程名称:" + Thread.currentThread().getName());
}
private static LazySingle Lazy ;
public static LazySingle getInstance(){ //调用该方法获取对象的时候,才去创建对象!
if (Lazy == null) {
Lazy = new LazySingle();
}
return Lazy ;
}
public static void main(String[] args) {
for (int i = 0; i < 10 ; i++) {
new Thread(() -> {
getInstance() ; //创建十个线程去测试!
}).start();
}
}
}
发现上述代码存在问题如下:
我们发现在多线程的情况下,会发生线程安全问题!
需要加锁,保证我们的线程安全问题 ,这里我们使用的是DCL(double check Lock)双检锁
public static LazySingle getInstance(){ //调用该方法获取对象的时候,才去创建对象!
if (Lazy == null) {
synchronized (LazySingle.class){
if (Lazy == null){
Lazy = new LazySingle();
}
}
}
return Lazy ;
}
看着结果是没问题了,但是在多线程情况下,由于new对象不是原子性操作,上述代码仍然存在指令重排的问题
我们先看一下new 的字节码指令:
// 通过这个复制的引用调用它的构造方法
21: invokespecial #4 // Method "<init>":()V
// 最开始的这个引用用来进行赋值操作
24: putstatic #2 // Field INSTANCE:Lcn/itcast/n5/Singleton;
此时我们的发生指令重排,先执行赋值操作,将空的实例对象返回(此时Lazy实例已经有值了),然后执行构造初始化对象!
就在我们的的初始化执行一半,线程t2过来了,发现Lazy不为null,执行 return Lazy;我们此时返回还是未被初始化的对象,所
以问题就此发生!!
通过volatile解决指令重排问题即可:
private static volatile LazySingle Lazy ;
对volatile修饰的变量进行写操作的时候,在写操作后加上内存屏障,使得写屏障之前的代码不会发生指令重排!
小结
饿汉式单例模式,当类加载的时候就直接实例化对象,因此不需要考虑线程安全问题。
懒汉与饿汉式如何选择?
本文主要给大家介绍Mysql数据库分库和分表方式(常用),涉及到mysql数据库相关知识,对mysql数据库分库分表相关知识感兴趣的朋友一起学习吧 1 分库 1.1 按照功能分库 按照功能进行
在当前对象由其他包含对象操作的系统中,当传递对当前对象的引用时,链接似乎一直在继续......没有任何结束(对于下面的代码,Car ->myCurrentComponent->myCar_Brake-
我有一个密码 UIAlertView,我们要求用户提供。我需要根据情况在不同的 View 上询问它,从 downloadViewController (用户下载数据后),当他们切换到他们的数据时(如果
我正在尝试编写一个函数,使得对于任何整数 x 的 P(x) 都有一个包含三个元素的列表,即平方、立方和 n 的四次方,但我仍然不知道如何组合然后制作一个函数,例如我有平方、立方体和 4 次幂函数下面是
关闭。这个问题需要更多 focused .它目前不接受答案。 关闭4年前。 锁定。这个问题及其答案是locked因为这个问题是题外话,但具有历史意义。它目前不接受新的答案或交互。 我能否列出一份常见的
Python 常用 PEP8 编码规范 代码布局 缩进 每级缩进用4个空格。 括号中使用垂直隐式缩进或使用悬挂缩进。 EXAMPLE: ?
关闭。这个问题需要更多focused .它目前不接受答案。 想改善这个问题吗?更新问题,使其仅关注一个问题 editing this post . 去年关闭。 Improve this questio
在经典 ui 中,您可以使用 xtype:cqinclude 包含来自不同路径的 rtePlugins,基本上为标准 RTE 插件创建一个位置,我如何在 Touch UI 中执行相同操作? 我尝试使用
在经典 ui 中,您可以使用 xtype:cqinclude 包含来自不同路径的 rtePlugins,基本上为标准 RTE 插件创建一个位置,我如何在 Touch UI 中执行相同操作? 我尝试使用
*strong text*我有多个网络应用程序使用了一些常见的依赖项,比如蒙戈连接器谷歌 Guava 乔达时间 我想到将它们从 webapp/WEB-INF/lib 中取出并放入一些 common-l
我正在编写一个 Web 服务器,我想知道哪些 HTTP 请求 header (由客户端发送)是最常见的,因此我应该重点实现。 目前,我只支持Accept 和Host。 最佳答案 不确定您的范围,但由于
我是一名优秀的程序员,十分优秀!