线程安全的恶汉单例实现(1)

unsingleton=new Singleton();

}

return unsingleton;

}

}




### 2.饿汉模式



> 线程安全



public class Singleton {

private static Singleton unsingleton=new Singleton();

private Singleton(){}

public static Singleton getInstance(){

return unsingleton;

}

}




调用



public class Test {

public static void main(String[] args) {

Singleton singleton1=Singleton.getInstance();

}

}




三.懒汉模式优化成线程安全

-------------



懒汉模式要变成线程安全的除了用饿汉模式之外,还有两种方法。



### 1.加synchronized关键字



> 此方法是最简单又有效的方法,不过对性能上会有所损失。比如两个线程同时调用这个实例,其中一个线程要等另一个线程调用完才可以继续调用。而线程不安全往往发生在这个实例在第一次调用的时候发生,当实例被调用一次后,线程是安全的,所以加**synchronized**就显得有些浪费性能。



public class Singleton {

private static Singleton unsingleton;

private Singleton(){}

public static synchronized Singleton getInstance(){

if(unsingleton==null){

unsingleton=new Singleton();

}

return unsingleton;

}

}




### 2.用"双重检查加锁"



> 上个方法说到,线程不安全往往发生在这个实例在第一次调用的时候发生,当实例被调用一次后,线程是安全的。那有没有方法只有在第一次调用的时候才用**synchronized**关键字,而第一次后就不用**synchronized**关键字呢?答案是当然有的,就是用**volatile**来修饰静态变量,保持其可见性。



public class Singleton {

private static volatile Singleton unsingleton;

private Singleton(){}

public static Singleton getInstance(){

if(unsingleton==null){

//只有当第一次访问的时候才会使用synchronized关键字

synchronized (Singleton.class){

unsingleton=new Singleton();

}

}

return unsingleton;

}

}




### 3\. 用"静态内部类"



静态内部类的优点是:外部类加载时并不需要立即加载内部类,内部类不被加载则不去初始化INSTANCE,故而不占内存。即当SingleTon第一次被加载时,并不需要去加载SingleTonHoler,只有当getInstance()方法第一次被调用时,才会去初始化INSTANCE,第一次调用getInstance()方法会导致虚拟机加载SingleTonHoler类,这种方法不仅能确保线程安全,也能保证单例的唯一性,同时也延迟了单例的实例化。



public class Singleton {

private Singleton() {

  }



  private static Singleton instatnce;



  private static class SingletonHolder {

    private static Singleton instatnce = new Singleton();

  }



  public static Singleton getInstance() {

    return SingletonHolder.instatnce;

  }

}



**那么,静态内部类又是如何实现线程安全的呢?首先,我们先了解下类的加载时机。**



**类加载时机:JAVA虚拟机在有且仅有的5种场景下会对类进行初始化。**



1.  遇到new、getstatic、setstatic或者invokestatic这4个字节码指令时,对应的java代码场景为:new一个关键字或者一个实例化对象时、读取或设置一个静态字段时(final修饰、已在编译期把结果放入常量池的除外)、调用一个类的静态方法时。

    

2.  使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没进行初始化,需要先调用其初始化方法进行初始化。

    

3.  当初始化一个类时,如果其父类还未进行初始化,会先触发其父类的初始化。

    


### 给大家的福利


**零基础入门**


对于从来没有接触过网络安全的同学,我们帮你准备了详细的学习成长路线图。可以说是最科学最系统的学习路线,大家跟着这个大的方向学习准没问题。


![](https://img-blog.csdnimg.cn/img_convert/95608e9062782d28f4f04f821405d99a.png)


同时每个成长路线对应的板块都有配套的视频提供:


![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/a91b9e8100834e9291cfcf1695d8cd42.png#pic_center)


因篇幅有限,仅展示部分资料

**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化资料的朋友,可以点击这里获取](https://bbs.csdn.net/topics/618540462)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值