当前位置: 首页 > news >正文

单例模式---JAVA

目录

“饿汉”模式

完整代码

“懒汉”模式

完整代码


单例模式:保证某个类在程序中只存在唯一一份实例, 而不会创建出多个实例。

单例模式可以通过实例创建的时间来分为两种:“饿汉”和“懒汉”模式。

“饿汉”模式

所谓的“饿汉”模式实则就是在类加载的时候创建出实例。

首先我们先创建一个类Singleton再在类中写一个静态私有的常量,而这个常量的值就是唯一对象的引用。

class Singleton{private static final Singleton singleton = new Singleton();
}

因为这个唯一对象是私有的所以还需要一个get方法。

public static Singleton getSingleton() {return singleton;
}

可是现在我们从别的地方还是可以直接new出这个类的其它实例,这个该怎么解决呢?我们只需要再写一个私有的构造方法就可以解决了。

private Singleton() {}

这就是一个简单的单例模式了(“饿汉”模式)

完整代码

class Singleton{private static final Singleton singleton = new Singleton();public static Singleton getSingleton() {return singleton;}private Singleton() {}
}

但是由于“饿汉”模式的实例是在类加载时就创建了,并没有考虑这个实例在代码中是否使用,这就有可能会导致代码中并没有用这个类可是你却已经创建了,这就会导致内存浪费,解决办法就是“懒汉”模式。

“懒汉”模式

“懒汉”模式是在线程的一次调用该类的get方法时进行唯一实例的创建。

先创建一个类,该类中有一个私有的类属性,该属性的值为null或唯一实例的引用。 

class Singleton{private static Singleton singleton = null;
}

为了保证实例的唯一性,将构造方法写为私有的。

private Singleton() {}

写一个get方法,该方法在第一次被调用时会创建出一个唯一实例。

public static Singleton getSingleton() {if (singleton == null) {singleton = new Singleton();}return singleton;
}

这个get方法在单线程中看是没有任何问题的,但是如果放在多线程代码中就会出现线程安全问题,例如如果出现以下的执行顺序那么就不是单例模式了。

解决办法就是加锁

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

但是此时代码又面临了一个效率问题,由于我们只有第一次调用get时才会创建实例才会出现线程安全问题,可是现在我们每次调用get方法都会进行加锁操作,而加锁就会有锁竞争从而导致代码效率过低的问题,解决方法就是再加一层 if 判断。

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

因为new操作很有可能触发指令重排序,所以为了防止编译器对其进行优化建议加上volatile

 

private static volatile Singleton singleton = null;

完整代码

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

http://www.lryc.cn/news/280759.html

相关文章:

  • maven管理使用
  • 如何在一个系统中同时访问异构的多种数据库
  • 半监督学习 - 半监督聚类(Semi-Supervised Clustering)
  • 实现STM32烧写程序-(3) Hex文件结构
  • 精品量化公式——“区域突破”,应对当下行情较好的主图看盘策略
  • 自然语言处理5——发掘隐藏规律 - Python中的关联规则挖掘
  • 【记录】重装系统后的软件安装
  • Android 13 - Media框架(31)- ACodec(七)
  • 快速了解VR全景拍摄技术运用在旅游景区的优势
  • 分布形态的度量_峰度系数的探讨
  • HCIP 重发布
  • FX图中的节点代表什么操作
  • 【Java 设计模式】创建型之单例模式
  • FlinkAPI开发之窗口(Window)
  • 【Unity】Joystick Pack摇杆插件实现锁四向操作
  • 29 旋转工具箱
  • WeNet2.0:提高端到端ASR的生产力
  • 第九部分 使用函数 (四)
  • 一文读懂「Prompt Engineering」提示词工程
  • 微信小程序(一)简单的结构及样式演示
  • 【设计模式】外观模式
  • 优先级队列(Priority Queue)
  • 12-桥接模式(Bridge)
  • Zookeeper+Kafka概述
  • 架构师 - 架构师是做什么的 - 学习总结
  • 全链路压测方案(一)—方案调研
  • c++关键字const
  • 分布式计算平台 Hadoop 简介
  • 系统学习Python——警告信息的控制模块warnings:常见函数-[warnings.warn]
  • 监听键盘事件vue3封装hooks