1 概念
一个类有且仅有一个实例,并且自行实例化向整个系统提供。
2 适用场景
一些资源管理器构件必须只有一个实例。
3 实现以及优缺点
单例优点:提供了对唯一实例的受控访问,由于在系统内存中只存在一个对象,因此可以节约系统资源。
单例缺点:在一定程度上违背了单一职责原则。
3.1 饿汉模式
优点:在类被初始化的时候就在内存中创建了对象,故线程安全。
缺点:空间换时间
package Singleton;
/**
* 饿汉模式
*/
public class SingletonOne {
private SingletonOne instance = new SingletonOne();
private SingletonOne() {
}
private SingletonOne getInstance() {
return instance;
}
}
3.2 懒汉模式
缺点:不用就不实例化,在方法被调用后才创建对象,在多线程下存在风险。为线程非安全方式。
package Singleton;
/**
* 懒汉模式
*/
public class SingletonTwo {
private SingletonTwo instance;
private SingletonTwo() {
}
private SingletonTwo getInstance() {
if (null == instance) {
return new SingletonTwo();
}
return instance;
}
}
3.3 线程安全的懒汉式
优点:线程安全。
缺点:由于每次调用都需要进行同步,而且大部分情况下实例是已经创建成功了,造成了不必要的同步开销,不建议用这种方式。
package Singleton;
/**
* 线程安全的懒汉式
*/
public class SingletonThree {
private SingletonThree instance;
private SingletonThree() {
}
private synchronized SingletonThree getInstance() {
if (null == instance) {
return new SingletonThree();
}
return instance;
}
}
3.4 双重检查模式的懒汉式DCL
该方法线程安全,声明为volatile实例对象,确保了多线程时的可见性,同时同步方法移步到代码块中,只有在该对象没有被实例化的时候才调用同步方法,节省了一部分开销。
特点:其一为用volatile修饰类实例对象;其二同步代码块。
package Singleton;
public class SingletonDcl {
//volatile修饰的值,在线程独有的工作内存中,线程直接和主内存交互,如果修改,则主内存立即可见。
private volatile SingletonDcl instance = null;
//私有构造器,使外部调用初始化时只能通过调用getInstance这个静态方法来获得实例。
private SingletonDcl() {
}
private SingletonDcl getInstance() {
if (null == instance) //首先进行非空判断。
{
synchronized (SingletonDcl.class)//对整个类进行加锁,限制当前对象只能被一个线程访问
{
return new SingletonDcl();
}
}
return instance;
}
}
3.5 静态内部类单例模式
特点:利用静态类只会加载一次的机制,由于在调用 SingletonHolder.instance 的时候,才会对单例进行初始化,故节省了内存开销。
package Singleton;
/**
* 线程安全静态内部类
*/
public class SingletonStatic {
private SingletonStatic() {
}
public static class SingletonHolder {
private static SingletonStatic instance = new SingletonStatic();
}
private SingletonStatic getInstance() {
return SingletonHolder.instance;
}
}
4 举例实践
4.1 JDK Runtime,饿汉模式
java是单进程,由于一个java程序启动一个jvm进程,一个jvm进程对应一个Runtime实例,使应用程序能够与其运行的环境相连接。
public class Runtime {
private static Runtime currentRuntime = new Runtime();
/**
* Returns the runtime object associated with the current Java application.
* Most of the methods of class <code>Runtime</code> are instance
* methods and must be invoked with respect to the current runtime object.
*
* @return the <code>Runtime</code> object associated with the current
* Java application.
*/
public static Runtime getRuntime() {
return currentRuntime;
}
}
4.2 Logger
5.扩展
问:为什有私有构造器
答:防止外部构造者直接实例化对象。
问:对象在实例化过程中的操作实例化空间如何计算
答:首先了解存储区域,请参考JVM读书笔记