1. 饿汉式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
// 饿汉式单例
public class Hungry {

// 可能会浪费空间
private byte[] data1 = new byte[1024*1024];
private byte[] data2 = new byte[1024*1024];
private byte[] data3 = new byte[1024*1024];
private byte[] data4 = new byte[1024*1024];

private Hungry(){

}

private final static Hungry HUNGRY = new Hungry();

public static Hungry getInstance(){
return HUNGRY;
}

}

2. DCL 懒汉式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
public class LazyMan {

private LazyMan(){
}

private volatile static LazyMan lazyMan;

// 双重检测锁模式的 懒汉式单例 DCL懒汉式
public static LazyMan getInstance(){
if (lazyMan==null){
synchronized (LazyMan.class){
if (lazyMan==null){
lazyMan = new LazyMan(); // 不是一个原子性操作
}
/**
* 1. 分配内存空间
* 2、执行构造方法,初始化对象
* 3、把这个对象指向这个空间
*
* 123
* 132 A线程执行132,处于3的位置,还未到2
* B线程进入,发现非空,执行return
// 此时lazyMan还没有完成构造,如果另一个线程B进来会发现已经分配了内存空间的lazyMan为非空,就会执行下面的return,但此时lazyMan还未初始化,这就会导致错误。所以需要在上面定义时加上volatile,防止指令重排导致此问题的发生。
*/
}
}
return lazyMan;
}

下面使用反射破坏DCL懒汉

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
// 懒汉式单例
public class LazyMan {

private static boolean qinjiang = false;

private LazyMan(){
synchronized (LazyMan.class){
if (qinjiang == false){
qinjiang = true;
}else {
throw new RuntimeException("不要试图使用反射破坏异常");//使用这些方法最终都无法阻止反射破坏
}
}
}

private volatile static LazyMan lazyMan;

// 双重检测锁模式的 懒汉式单例 DCL懒汉式
public static LazyMan getInstance(){
if (lazyMan==null){
synchronized (LazyMan.class){
if (lazyMan==null){
lazyMan = new LazyMan(); // 不是一个原子性操作
}

}
}
return lazyMan;
}

// 反射!
public static void main(String[] args) throws Exception {
// LazyMan instance = LazyMan.getInstance();

Field qinjiang = LazyMan.class.getDeclaredField("qinjiang");
qinjiang.setAccessible(true);

Constructor<LazyMan> declaredConstructor =
LazyMan.class.getDeclaredConstructor(null);
declaredConstructor.setAccessible(true);
LazyMan instance = declaredConstructor.newInstance();

qinjiang.set(instance,false);

LazyMan instance2 = declaredConstructor.newInstance();

System.out.println(instance);
System.out.println(instance2);
}

}


3.其他

1
2
3
4
5
6
7
8
9
10
11
12
13
14
// 静态内部类
public class Holder {
private Holder(){
}

public static Holder getInstace(){
return InnerClass.HOLDER;
}

public static class InnerClass{
private static final Holder HOLDER = new Holder();
}

}

单例不安全,可以用反射破坏!!

枚举也就一种单例,但枚举不能用反射破坏

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
// enum 是一个什么? 本身也是一个Class类
public enum EnumSingle {

INSTANCE;

public EnumSingle getInstance(){
return INSTANCE;
}

}

class Test{

public static void main(String[] args) throws NoSuchMethodException,
IllegalAccessException, InvocationTargetException, InstantiationException {
EnumSingle instance1 = EnumSingle.INSTANCE;
Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor();// NoSuchMethodException: com.kuang.single.EnumSingle.<init>(),idea等工具反编译的字节码文件显示枚举类有空参构造器,用反射获取显示没有空参构造器,用jad反编译得知,枚举使用的是带两个参数(String,int)的构造器
// Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor(String.class,int.class);//使用这个构造器可以继续往下执行,直到返回枚举中定义的“无法使用反射破坏枚举”的异常信息
declaredConstructor.setAccessible(true);
EnumSingle instance2 = declaredConstructor.newInstance();


System.out.println(instance1);
System.out.println(instance2);

}

}

如下,使用idea和javap等工具都获取到的是枚举中用的是空参构造器,事实却并非如此,使用更专业的jad反编译可得到正确的带两个参数(String,int) 的构造器。

image-20210328121136110

枚举类型的最终反编译源码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
public final class EnumSingle extends Enum
{

public static EnumSingle[] values()
{
return (EnumSingle[])$VALUES.clone();
}

public static EnumSingle valueOf(String name)
{
return (EnumSingle)Enum.valueOf(com/kuang/single/EnumSingle, name);
}

private EnumSingle(String s, int i) //正确的构造器!!
{
super(s, i);
}

public EnumSingle getInstance()
{
return INSTANCE;
}

public static final EnumSingle INSTANCE;
private static final EnumSingle $VALUES[];

static
{
INSTANCE = new EnumSingle("INSTANCE", 0);
$VALUES = (new EnumSingle[] {
INSTANCE
});
}
}