
package com.company.java.oop.instance;
/**
* 如何设计,才能保证如下类的实例对象
* 在内存中只有一份?
* 1)让外界从一个池中去取对象(通过池保证类的实例在池中一份)
* 2)让外界直接通过类的静态方法获取类内部创建的一个实例。
*/
class Singleton01{//应用场景:小对象
static Singleton01 instance=new Singleton01();
private Singleton01() {}
public static Singleton01 getInstance() {
return instance;
}
}
class Singleton02{//适合场景:大对象,单线程
private Singleton02() {
System.out.println("Singleton02()");
}
private static Singleton02 instance;
/**
* 线程不安全?导致线程不安全的原因是什么?
* 1)多个线程并发执行
* 2)多个线程有共享数据集(例如instance)
* 3)多个线程在共享数据上的操作不是原子操作。
* @return
*/
public static Singleton02 getInstance() {
if(instance==null) {
instance=new Singleton02();
}
return instance;
}
}
class Singleton04{//适合场景:大对象,线程不多,性能要求不高
private Singleton04() {
System.out.println("Singleton04()");
}
/**volatile 关键字的作用
* 1)禁止指令重排序(JVM对java程序进行编译时会有指令重排序的现象)
* 2)保证可见性(多CPU,一个线程的对某个共享变量的修改对另一个线程是可见的)
* 3)但不保证原子性(synchronized可以保证)。
*/
private static volatile Singleton04 instance;
/**
* 优化:在保证线程安全的情况下尽量减少线程阻塞。
* @return
*/
public static Singleton04 getInstance() {
if(instance==null) {
synchronized (Singleton04.class) {
if(instance==null) {
instance=new Singleton04();
//1.分配空间
//2.对象属性初始化
//3.执行构造方法
//4.将对象赋值给instance变量
}
}
}
return instance;
}
}
public class TestInstanceObject04 {
public static void main(String[] args) {
Thread t1=new Thread() {
public void run() {
Singleton03.getInstance();
Singleton03.getInstance();
Singleton03.getInstance();
};
};
Thread t2=new Thread() {
public void run() {
Singleton03.getInstance();
Singleton03.getInstance();
Singleton03.getInstance();
};
};
t1.start();
t2.start();
}
}