private Singleton03(){}
public static Singleton03 getInstance() {
if(instance == null) {
synchronized (Singleton03.class) {
if (instance == null) {
instance = new Singleton03();
}
}
}
return instance;
}
}
使用到了volatile机制。这个是第二种方式的升级版,俗称双重检查锁定。既保证了效率,又保证了安全。
4)饿汉模式
public class Singleton03 {
private static Singleton03 instance = new Singleton03();
/**
- 私有构造方法
*/
private Singleton03(){}
public static synchronized Singleton03 getInstance() {
return instance;
}
}
**这种基于类加载机制避免了多线程的同步问题,初始化的时候就给装载了。但却没了懒加载的效果。
这也是最简单的一种实现。**
5)静态内部类
public class Singleton04 {
// 静态内部类
private static class SingletonHolder {
private static final Singleton04 INSTANCE = new Singleton04();
}
/**
- 私有构造方法
*/
private Singleton04(){}
public static Singleton04 getInstance() {
return SingletonHolder.INSTANCE;
}
}
**这种方式当Singleton04
类被加载时,其内部类并不会被加载,所以单例类INSTANCE
不会被初始化。
只有显式调用getInstance
方法时,才会加载SingletonHolder
,从而实例化INSTANCE
。
由于实例的建立是在类加载时完成,所以天生线程安全。因此兼备了懒加载和线程安全的特性。**
6)枚举(号称最好)
public enum EnumSingleton01 {
INSTANCE;
public void doSomething() {
System.out.println(“doSomething”);
}
}
模拟数据库链接:
public enum EnumSingleton02 {
INSTANCE;
private DBConnection dbConnection = null;
private EnumSingleton02() {
dbConnection = new DBConnection();
}
public DBConnection getConnection() {
return dbConnection;
}
}
**这种方式是Effective Java
作者Josh Bloch
提倡的方式,它不仅能避免多线程同步问题,
而且还能防止反序列化重新创建新的对象。**
2、为什么说枚举方法是最好的?
前5种方式实现单例都有如下3个特点:
-
构造方法私有化
-
实例化的变量引用私有化
-
获取实例的方法共有
首先,私有化构造器并不保险。因为它抵御不了反射攻击
,其次就是序列化重新创建新对象。下面来进行验证。
1) 反射验证
@Test
public void reflectTest() throws Exception {
Singleton03 s = Singleton03.getInstance();
// 拿到所有的构造函数,包括非public的
Constructor constructor = Singleton03.class.getDeclaredConstructor();
constructor.setAccessible(true);
// 构造实例
Singleton03 reflection = constructor.newInstance();
System.out.println(s);
System.out.println(reflection);
System.out.println(s == reflection);
}
输出结果:
org.yd.singleton.Singleton03@61e4705b
org.yd.singleton.Singleton03@50134894
false
再看看枚举类的测试
@Test
public void reflectEnumTest() throws Exception {
EnumSingleton01 s = EnumSingleton01.INSTANCE;
// 拿到所有的构造函数,包括非public的
Constructor constructor = EnumSingleton01.class.getDeclaredConstructor();
constructor.setAccessible(true);
// 构造实例
EnumSingleton01 reflection = constructor.newInstance();
System.out.println(s);
System.out.println(reflection);
System.out.println(s == reflection);
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新
如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
感受:
其实我投简历的时候,都不太敢投递阿里。因为在阿里一面前已经过了字节的三次面试,投阿里的简历一直没被捞,所以以为简历就挂了。
特别感谢一面的面试官捞了我,给了我机会,同时也认可我的努力和态度。对比我的面经和其他大佬的面经,自己真的是运气好。别人8成实力,我可能8成运气。所以对我而言,我要继续加倍努力,弥补自己技术上的不足,以及与科班大佬们基础上的差距。希望自己能继续保持学习的热情,继续努力走下去。
也祝愿各位同学,都能找到自己心动的offer。
分享我在这次面试前所做的准备(刷题复习资料以及一些大佬们的学习笔记和学习路线),都已经整理成了电子文档
感受:
其实我投简历的时候,都不太敢投递阿里。因为在阿里一面前已经过了字节的三次面试,投阿里的简历一直没被捞,所以以为简历就挂了。
特别感谢一面的面试官捞了我,给了我机会,同时也认可我的努力和态度。对比我的面经和其他大佬的面经,自己真的是运气好。别人8成实力,我可能8成运气。所以对我而言,我要继续加倍努力,弥补自己技术上的不足,以及与科班大佬们基础上的差距。希望自己能继续保持学习的热情,继续努力走下去。
也祝愿各位同学,都能找到自己心动的offer。
分享我在这次面试前所做的准备(刷题复习资料以及一些大佬们的学习笔记和学习路线),都已经整理成了电子文档
[外链图片转存中…(img-m3i8I2lH-1711646303695)]