1 线程不安全的实现方法
首先介绍java中最基本的单例模式实现方式,我们可以在一些初级的java书中看到。这种实现方法不是线程安全的,所以在项目实践中如果涉及到线程安全就不会使用这种方式。但是如果不需要保证线程安全,则这种方式还是不错的,因为所需要的开销比较小。下面是具体的实现代码:
public Class Singleton
{
private static Singleton instance = null;
private Singleton(){}
public static Singleton getInstance()
{
if( instance == null)
instance = new Singleton ();
return instance;
}
}
我们说过这种实现方式不是thread-safe的,那么可以把上面的方法变成线程安全的吗?当然可以,在方法getInstance()上加上synchronized修饰符就可以实现方法的同步了。但是这样系统开销会很大。具体代码如下:
public Class Singleton
{
private static Singleton instance = null;
private Singleton(){}
public static synchronized Singleton getInstance()
{
if( instance == null)
instance = new Singleton ();
return instance;
}
}
每次有线程调用getInstance()方法,都需要同步判断。这显然不是最好的选择,下面将会陆续介绍几种thread-safe的方法。
2 两种lazy loaded thread-safe的单例模式实现方式
1) DCL (double checked locking 实现法)
double checked locking ,顾名思义,就是双检查法,检查实例INSTANCE是否为null或者已经实例化了。下面是具体的实现代码:
public class DoubleCheckedLockingSingleton{
private volatile DoubleCheckedLockingSingleton INSTANCE;
private DoubleCheckedLockingSingleton(){}
public DoubleCheckedLockingSingleton getInstance(){
if(INSTANCE == null){
synchronized(DoubleCheckedLockingSingleton.class){
//double checking Singleton instance
if(INSTANCE == null){
INSTANCE = new DoubleCheckedLockingSingleton();
}
}
}
return INSTANCE;
}
}
这种方法也很好理解,我们可以看到有两次对instance是否为null的判断:如果第一次判断不为空,则直接返回实例就可以了;如果instance为空,则进入同步代码块再进行null值判断,再选择是否实例化。第一个null判断可以减少系统的开销。在实际项目中做过多线程开发的都应该知道DCL。
2) lazy initialization holder class 模式实现法
下面是这种方法的实现代码:
public class Singleton {
/**
* 类级的内部类,也就是静态的成员式内部类,该内部类的实例与外部类的实例
* 没有绑定关系,而且只有被调用到才会装载,从而实现了延迟加载
*/
private static class SingletonHolder{
/**
* 静态初始化器,由JVM来保证线程安全
*/
private static Singleton instance = new Singleton();
}
/**
* 私有化构造方法
*/
private Singleton(){
}
public static Singleton getInstance(){
return SingletonHolder.instance;
}
}
当getInstance方法第一次被调用的时候,它第一次读取SingletonHolder.instance,导致SingletonHolder类得到初始化;而这个类在装载并被初始化的时候,会初始化它的静态域,从而创建Singleton的实例,由于是静态的域,因此只会被虚拟机在装载类的时候初始化一次,并由虚拟机来保证它的线程安全性。这个模式的优势在于,getInstance方法并没有被同步,并且只是执行一个域的访问,因此延迟初始化并没有增加任何访问成本。
关于延迟初始化(lazy loaded)
“除非绝对必要,否则就不要延迟初始化”。延迟初始化是一把双刃剑,它降低了初始化类或者创建实例的开销,却增加了访问被延迟初始化的域的开销,考虑到延迟初始化的域最终需要初始化的开销以及域的访问开销,延迟初始化实际上降低了性能。
3 静态工厂实现法
因为单例是静态的final变量,当类第一次加载到内存中的时候就初始化了,其thread-safe性由JVM来负责保证。值得注意的是这个实现方式不是lazy-loadedd的。 具体实现代码如下:
public class Singleton{
//initailzed during class loading
private static final Singleton INSTANCE = new Singleton();
private Singleton(){}
public static Singleton getSingleton(){
return INSTANCE;
}
}
4 枚举实现单例(Enum Singleton)
枚举单例(Enum Singleton)是实现单例模式的一种新方式,枚举这个特性是在Java5才出现的,在《Effective Java》一书中有介绍这个特性。下面是这种方法的具体实现代码:
public enum Singleton {
INSTANCE("hello") {
public void someMethod() {
// . . .
}
};
private String name;
private void PrintName(){System.out.println(name);}
protected abstract void someMethod();
}
你可以通过Singleton.INSTANCE来访问该单示例变量。默认枚举实例的创建是线程安全的,但是在枚举中的其他任何方法由程序员自己负责。如果你正在使用实例方法,那么你需要确保线程安全(如果它影响到其他对象的状态的话)。传统单例存在的另外一个问题是一旦你实现了序列化接口,那么它们不再保持单例了,但是枚举单例,JVM对序列化有保证。枚举实现单例的好处:有序列化和线程安全的保证,代码简单。
分享到:
相关推荐
Lan仿朋友圈系统开源,可用于表白墙等微商相册,商品图册等.rarLan仿朋友圈系统开源,可用于表白墙等微商相册,商品图册等.rar
比如异步进行-Thread,安全句柄-CHandle,资源守卫-Guard,XML解析-rapidxml,以及其他注册表、文件基础操作。用于更加高效、安全的进行C++开发。温馨提示:至少需要支持C++0x标准的编译器。
集团企业IT技术架构规划方案qy.pptx
本Bash脚本用于自动化管理Java JAR应用的启动、停止及监控。首先检查JAR进程是否在运行,如在运行则安全终止。随后,使用预设的Java参数启动JAR文件,并将输出和错误日志重定向至日志文件。启动后,脚本持续监控JAR进程状态,确保其在预设时间内成功启动。本脚本提供了灵活的配置和错误处理机制,为Java应用的运维管理带来了便捷与可靠性。
基于价值认同的需求侧电能共享分布式交易策略matlab源码+项目说明+详细注释.zip # Supports_for_EPC 电力建设论文《基于价值认同的需求侧电能共享分布式交易策略》的支撑文件 ————————————学术交流———————————————————— 本论文提出了一种电能共享市场交易机制,利用一致性协议实现产消者之间关于价值的认同,实现了社会福利的最大化。 main_CA.m:针对10个产消者之间的电能共享,利用一致性算法完成市场的分布式出清 Pareto_analysis.m:通过两个产消者分析了电能共享市场的广义纳什均衡与市场效率。 运行环境: MATLAB R2014a YALMIP GUROBI 需要注意的是:若程序运行错误,请认真检查是否安装了YALMIP和GUROBI求解器。 ......
2024-2030全球及中国太阳能直流冰柜行业研究及十五五规划分析报告
springcodespringcodespringcodespringcode
2023年前10大汽车用电流检测放大器企业占据全球86%的市场份额
在STM32单片机上调试CAN总线并与其上位机通信时,需要遵循一系列的步骤和策略,以确保通信的可靠性和效率。以下是一些关键步骤和考虑因素: 硬件配置:确保STM32的CAN接口引脚正确连接到CAN收发器,并且CAN总线上的每个节点都通过120Ω的终端电阻正确终止。 波特率设置:所有CAN网络节点必须使用相同的波特率进行通信。波特率可以通过STM32的CAN外设的位时序寄存器(CAN_BTR)来设置。 初始化CAN外设:使用STM32的HAL库或标准库函数初始化CAN外设,包括设置工作模式、位时序、过滤器等。 过滤器配置:配置CAN过滤器以确定哪些消息ID可以被接收或发送。 中断配置:为了处理接收到的消息,通常需要配置CAN接收中断,并在中断服务例程(ISR)中处理消息。 发送和接收消息:实现CAN发送和接收消息的函数,使用HAL_CAN_Transmit_IT和HAL_CAN_Receive_IT等函数进行异步通信。 错误处理:实现错误处理机制,以应对可能的通信错误,如消息冲突、总线错误等。 回环测试:在只有一个STM32板的情况下,可以使用CAN的回环模式进行测试,以
2024年中国克拉霉素混悬液行业研究报告
EasyAnimate生成
Thinkphp内核仿918国际导航简洁模板网址导航源码.rarThinkphp内核仿918国际导航简洁模板网址导航源码.rar
基于matlab实现潮流计算的程序,可以帮助学习潮流计算。使用matlab编程,程序简单。好好学习潮流计算.rar
视觉毫米波简单后融合方法 视觉目标检测 毫米波目标检测
基于Reactor模式,支持单线程、多线程Reactor,也支持UDP服务
机械设计螺旋输送机sw17非常好的设计图纸100%好用.zip
有30w条数据
本Bash脚本用于自动化管理Java JAR应用的启动、停止及监控。首先检查JAR进程是否在运行,如在运行则安全终止。随后,使用预设的Java参数启动JAR文件,并将输出和错误日志重定向至日志文件。启动后,脚本持续监控JAR进程状态,确保其在预设时间内成功启动。本脚本提供了灵活的配置和错误处理机制,为Java应用的运维管理带来了便捷与可靠性。
嗨淘V12刷任务点赞系统源码+手动派单版本.rar嗨淘V12刷任务点赞系统源码+手动派单版本.rar
机械设计一次性手套机sw2016非常好的设计图纸100%好用.zip