3.2.1 饿汉方式
实现代码:
public class DataSourceSingleton {
// 1.提供私有的构造方法(防止外部直接new对象)
private DataSourceSingleton() {}
// 2.创建一个私有的属性对象
private static DataSourceSingleton dataSource = new DataSourceSingleton();
// 3.提供公共对外的单例对象
public static DataSourceSingleton getInstance() {
return dataSource;
}
}
3.2.2 懒汉方式
(1)单线程版本(非线程安全的):
public class DataSourceSingleton2 {
// 1.提供私有的构造方法(防止外部直接new对象)
private DataSourceSingleton2() {}
// 2.创建一个私有的属性对象
private static DataSourceSingleton2 dataSource = null;
// 3.提供公共对外的单例对象
public static DataSourceSingleton2 getInstance() {
if (dataSource == null) {
// 第一次访问
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
dataSource = new DataSourceSingleton2();
}
return dataSource;
}
}
测试:
public class DataSourceTest {
public static void main(String[] args) {
Thread t1 = new Thread(() -> {
System.out.println(DataSourceSingleton2.getInstance());
});
Thread t2 = new Thread(() -> {
System.out.println(DataSourceSingleton2.getInstance());
});
Thread t3 = new Thread(() -> {
System.out.println(DataSourceSingleton2.getInstance());
});
t1.start();
t2.start();
t3.start();
}
}
测试结果:
可以看到产生了三个实例,而非一个,线程不安全。这种方式只能在单线程下使用。如果在多线程下,一个线程进入了if (dataSource == null)判断语句块,还未来得及往下执行,另一个线程也通过了这个判断语句,这时便会产生多个实例。所以在多线程环境下不可使用这种方式。
(2)线程安全改进版一
public class DataSourceSingleton2 {
// 1.提供私有的构造方法(防止外部直接new对象)
private DataSourceSingleton2() {}
// 2.创建一个私有的属性对象
private static DataSourceSingleton2 dataSource = null;
// 3.提供公共对外的单例对象
public synchronized static DataSourceSingleton2 getInstance() {
if (dataSource == null) {
// 第一次访问
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
dataSource = new DataSourceSingleton2();
}
return dataSource;
}
}
测试结果:
可以看到只创建了一个实例。但是这种方式虽然线程安全,但是直接给全局加锁,效率太低了。不推荐使用。
(3)线程安全改进版二
public class DataSourceSingleton3 {
// 1.提供私有的构造方法(防止外部直接new对象)
private DataSourceSingleton3() {
}
// 2.创建一个私有的属性对象
private static volatile DataSourceSingleton3 dataSource = null;
// 3.提供公共对外的单例对象
public static DataSourceSingleton3 getInstance() {
if (dataSource == null) {
synchronized (DataSourceSingleton3.class) {
dataSource = new DataSourceSingleton3();
}
}
return dataSource;
}
}
测试结果:
可以看到这种方式也存在线程安全问题,假如一个线程进入了if (dataSource == null)判断语句块,还未来得及往下执行,另一个线程也通过了这个判断语句,这时便会产生多个实例。不可用。
(4)线程安全改进版三
public class DataSourceSingleton4 {
// 1.提供私有的构造方法(防止外部直接new对象)
private DataSourceSingleton4() {}
// 2.创建一个私有的属性对象
private static DataSourceSingleton4 dataSource = null;
### Java核心架构进阶知识点
面试成功其实都是必然发生的事情,因为在此之前我做足了充分的准备工作,不单单是纯粹的刷题,更多的还会去刷一些Java核心架构进阶知识点,比如:JVM、高并发、多线程、缓存、Spring相关、分布式、微服务、RPC、网络、设计模式、MQ、Redis、MySQL、设计模式、负载均衡、算法、数据结构、kafka、ZK、集群等。而这些也全被整理浓缩到了一份pdf——《Java核心架构进阶知识点整理》,全部都是精华中的精华,本着共赢的心态,好东西自然也是要分享的
![image](https://img-blog.csdnimg.cn/img_convert/3a0ac46dffa976ced181f1297819a200.webp?x-oss-process=image/format,png)
![image](https://img-blog.csdnimg.cn/img_convert/9c088935922f94f96707346e0228e107.webp?x-oss-process=image/format,png)
![image](https://img-blog.csdnimg.cn/img_convert/6db0c0199edd54597fa90775ddf4a6d8.webp?x-oss-process=image/format,png)
内容颇多,篇幅却有限,这就不在过多的介绍了,大家可根据以上截图自行脑补
ing相关、分布式、微服务、RPC、网络、设计模式、MQ、Redis、MySQL、设计模式、负载均衡、算法、数据结构、kafka、ZK、集群等。而这些也全被整理浓缩到了一份pdf——《Java核心架构进阶知识点整理》,全部都是精华中的精华,本着共赢的心态,好东西自然也是要分享的
[外链图片转存中...(img-0KcTYLmY-1714355153067)]
[外链图片转存中...(img-4vU6OPdy-1714355153068)]
[外链图片转存中...(img-oBDzJN0d-1714355153068)]
内容颇多,篇幅却有限,这就不在过多的介绍了,大家可根据以上截图自行脑补
> **本文已被[CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】](https://bbs.csdn.net/topics/618154847)收录**