一、单例模式构造方法一般是私有的,其他类不能直接调用构造方法
1.饿汉式单例
单例模式的成员变量和构造方法都是私有的
//当类被加载时创建对象,线程安全的,未使用就被创建可能会造成资源浪费
public class Test{
//成员变量是私有、静态的
//饿汉式单例的成员变量直接初始化,和懒汉式不同,懒汉式是null
private static Test test=new Test();
//构造方法是私有的
private Test(){
}
public static Test getInstance(){
return test;
}
}
2.懒汉式单例
//按需创建,但是多线程容易创建出多个对象
public class Test {
//静态方法内部变量也是要是静态的,且不进行初始化操作
//volatile 可见行,禁止JVM指令重排序
private volatile static Test test;
//构造方法要是私有的
private Test(){
}
public static Test getInstance(){
if (test==null){
synchronized(Test.class){
//多线程下可能同时多个线程阻塞在synchronized处,
//若不判断null则可能会创建多个test实例
if(test==null){
//使用volatile的原因:
//此处可能会发生JVM的指令重排序
//new操作的具体步骤是JVM先创建一个堆内存空间用于存放对象实例,
//此时会把该堆内存空间指向test引用,但是对象实例还没有完成初始化所以内存空间中是
空的,test引用只指向了一个空的内存空间地址。
//此时第一个线程的synchronized执行完毕,第二个线程进入synchronized执行。
//但是此时test引用已经指向了空内存空间所以引用不为 null然后直接返回
//最后在调用的时候就会发生空指针异常
//所以此处需要禁止重排序,使用volatile
test=new Test();
}
}
}
return test;
}
}
3. 第三种单例模式的写法(也是懒汉式),利用类加载
public class Test{
private Test(){
}
//此处用内部类维护单例,JVM在类加载的时候是互斥的,可以保证线程安全性
private static class SingletonFactory{
private static Test test=new Test();
}
public static Test getInstance(){
return SingletonFactory.test;
}
}