当前位置: 首页 > 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;}
}

相关文章:

  • 环信IM Demo登录方式如何修改为自己项目的?
  • 代码随想录刷题第四十三天| 1049. 最后一块石头的重量 II ● 494. 目标和 ● 474.一和零
  • Java开发+Intellij-idea+Maven+工程构建
  • Mysql in查询优化
  • SpingBoot的项目实战--模拟电商【5.沙箱支付】
  • IO进程线程Day6
  • springboot git配置文件自动刷新失败问题排查
  • IDEA UML图
  • C语言之素数进化论
  • .net6解除文件上传限制。Multipart body length limit 16384 exceeded
  • [论文阅读]4DRadarSLAM: A 4D Imaging Radar SLAM System for Large-scale Environments
  • Flutter中的Tree
  • 力扣188. 买卖股票的最佳时机 IV
  • cissp 第10章 : 物理安全要求
  • PHP中excel带图片数据导入
  • 收藏网友的 源程序下载网
  • AWS实战 - 利用IAM对S3做访问控制
  • CAP理论的例子讲解
  • css选择器
  • CSS选择器——伪元素选择器之处理父元素高度及外边距溢出
  • gops —— Go 程序诊断分析工具
  • js算法-归并排序(merge_sort)
  • leetcode388. Longest Absolute File Path
  • Nodejs和JavaWeb协助开发
  • Rancher-k8s加速安装文档
  • 闭包,sync使用细节
  • 测试开发系类之接口自动化测试
  • 程序员最讨厌的9句话,你可有补充?
  • 给Prometheus造假数据的方法
  • 基于Javascript, Springboot的管理系统报表查询页面代码设计
  • 理清楚Vue的结构
  • 排序(1):冒泡排序
  • 前端每日实战:61# 视频演示如何用纯 CSS 创作一只咖啡壶
  • 一个SAP顾问在美国的这些年
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • # include “ “ 和 # include < >两者的区别
  • $$$$GB2312-80区位编码表$$$$
  • (70min)字节暑假实习二面(已挂)
  • (delphi11最新学习资料) Object Pascal 学习笔记---第5章第5节(delphi中的指针)
  • (NSDate) 时间 (time )比较
  • (附源码)springboot码头作业管理系统 毕业设计 341654
  • (规划)24届春招和25届暑假实习路线准备规划
  • (黑马C++)L06 重载与继承
  • (六)c52学习之旅-独立按键
  • (转)setTimeout 和 setInterval 的区别
  • (转)创业的注意事项
  • (转)大道至简,职场上做人做事做管理
  • (转)真正的中国天气api接口xml,json(求加精) ...
  • .“空心村”成因分析及解决对策122344
  • .Net CF下精确的计时器
  • .NET/C# 获取一个正在运行的进程的命令行参数
  • .NET值类型变量“活”在哪?
  • ??myeclipse+tomcat
  • @font-face 用字体画图标
  • @RequestBody详解:用于获取请求体中的Json格式参数