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

编程(40)----------单例模式

在简单总结单例模式之前, 需要了解一下背景知识-----为何会有单例模式?

想象一个这样的场景, 打游戏的时候, 尝试很多次, 都未通关. 这种情况下是否会考虑查一下攻略? 一个好的攻略甚至可能连每一关的每一个场景由多少只怪物都说的清清楚楚. 再比如, 在以前上学的时候, 为了方便理解和使用一些解题技巧, 是不是会有一些口诀? 例如, 奇变偶不变, 符号看象限. 亦或者,为了能够下棋下的更好, 可能会尝试去看看棋谱. 棋谱里会把常见的开局解法给一一列举出来.

单例模式也是一个道理. 就跟上述的一样, 是为了方便代码的规范所做出的一些明确的规定,或者说代码的写法. 以此提高写代码的效率. 这些模式种类很多, 单例模式只是其中的一个. 顾名思义,单例模式就是只允许创建一个实例的模式.

代码写法有两种: 饿汉模式和懒汉模式

//饿汉模式
class Singleton{//先将实例创建出来private static Singleton instance = new Singleton();//若要使用唯一实例, 通过统一的方法public static Singleton getInstance(){return instance;}//将构造方法设为私有, 则无法在该类外使用该构造方法private Singleton(){}
}public class demo3 {public static void main(String[] args) {}
}
//懒汉模式
class Singleton{//先将实例创建出来private static Singleton instance = null;//若要使用唯一实例, 通过统一的方法public static Singleton getInstance(){if (instance == null){instance = new Singleton();}return instance;}//将构造方法设为私有, 则无法在该类外使用该构造方法private Singleton(){}
}public class demo3 {public static void main(String[] args) {}
}

代码本身并不复杂, 涉及到的也是一些简单的Java语法. 但是, 存在一个问题: 这两种模式是否线程安全?

在此之前, 得清楚衡量线程是否安全, 其中一个重要的指标就是, 是否该代码或者说线程只进行了读操作. 若既有读操作又有其他操作, 如写改操作. 那这个线程多半是不安全的.

再回到这俩个模式的代码中. 不难看出懒汉模式存在一个if语句的判定, 这其实就是读操作. 要先读入才能进行判定改写, 因此懒汉模式线程不安全. 反之, 饿汉模式只有读操作, 是安全的.

基于以上的结论, 假设要在多线程中使用懒汉模式. 就必须将其进行优化.

首先就从判定改写入手. 在多线程中, 由于读入, 判定, 改写的操作不具有原子性, 是分开的操作. 很有可能一个线程刚判定完, 为null, 另一个线程也开始判定, 结果也为null. 在这种情况下二者都会依照代码在此基础上new一个对象. 但是这实际上是违背了单例模式的规则. 因此要对整个过程进行加锁操作.

  synchronized (SingletonLazy.class) {if (instance == null) {instance = new SingletonLazy();}}

其次, 在线程很多的情况下, 需要每次都进行if判定是否为空吗? 很明显不需要, 单例模式只存在一个实例. 也就是说无需每次都加锁进行if判定, 效率过低. 当已经有某个线程new了一个对象以后,直接返回对象即可, 因此可以将代码进行再次优化:

public static SingletonLazy getSingletonLazy(){if (instance == null) {synchronized (SingletonLazy.class) {if (instance == null) {instance = new SingletonLazy();}}}return instance;}

除此以外, 当线程非常多且代码没有什么变化的时候, 系统自身会优化, 即发生内存可见性问题. 还有可能发生代码执行顺序调换的问题. 为解决以上问题, 使用volatile关键字一次性解决. 因此综合以上优化, 懒汉模式的多线程代码就差不多可以写出来了:

//懒汉模式
class SingletonLazy{private volatile static SingletonLazy instance = null;public static SingletonLazy getSingletonLazy(){if (instance == null) {synchronized (SingletonLazy.class) {if (instance == null) {instance = new SingletonLazy();}}}return instance;}private SingletonLazy(){}
}public class demo4 {public static void main(String[] args) {}
}

------------------------------最后编辑于2023.6.1晚上八点左右

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

相关文章:

  • Java开发 - 让你少走弯路的Redis主从实现单节点哨兵模式
  • Java的Atomic原子类
  • 离线语音控制新方案,NRK3303语音识别芯片在智能风扇的应用
  • 在树莓派3B+上安装Pytorch1.7
  • Java性能权威指南-总结4
  • c语言全局变量和局部变量问题汇总
  • 14.3:给定一个由字符串组成的数组strs,必须把所有的字符串拼接起来,返回所有可能的拼接结果中字典序最小的结果
  • C++ 项目实战:跨平台的文件与视频压缩解压工具的设计与实现
  • C和指针(二)数据
  • PyTorch基础学习(一)
  • chatgpt赋能python:Python代做:让您的网站更友好的SEO利器
  • 2022年都快结束了,还有人不会安卓录屏?在安卓上录制屏幕的的实现方式
  • px rem em rpx 区别 用法
  • 忆享聚焦|ChatGPT、AI、网络数字、游戏……近期热点资讯一览
  • [Daimayuan] 树(C++,动态规划,01背包方案数)
  • 如何选择源代码加密软件
  • TO-B类软件产品差异化
  • 设计模式之美-实战一(上):业务开发常用的基于贫血模型的MVC架构违背OOP吗?
  • ChatGPT如何训练自己的模型
  • springboot使用线程池的实际应用(一)
  • ESP-8266学习笔记
  • Java泛型简单的使用
  • 深度探索:Qt CMake工程编译后的自动打包策略
  • 2.7 编译型和解释型
  • 校园网自动登陆(河南科技学院)
  • C++11 override和final关键字
  • kafka的log存储解析
  • 4.文件系统
  • Shell脚本case in esac分支语句应用
  • 【线性dp必学四道题】线性dp四道经典例题【最长上升子序列】、【最长公共子序列】、【最长公共上升子序列(maxv的由来)】【最长公共子串】