在学习单例模式前,不妨问自己几个问题:单例模式是怎么来的,单例模式怎么去用?
单例模式是怎么来的?
这就从设计模式起源开始,他是在实际实践中遇到类似情况可以通用经验所得到的总结,一般在其他模块或者方法多次调用类对象,也就是公共模块,用单例模式可以减少内存的消耗。
单例模式怎么去用?
那这个容易,然后很快不到一分钟写完了。
1 public class singleTonEx01 {
2
3 private static singleTonEx01 singleTon;
4
5 public static void main(String args[]) {
6 singleTonEx01 test1=singleTonEx01.getInstance();
7 test1.CommonMethod();
8 }
9
10 // 防止使用new构造函数实例化对象
11 private singleTonEx01() {
12
13 }
14
15 public static singleTonEx01 getInstance() {
16 if(null==singleTon)
17 singleTon=new singleTonEx01();
18 return singleTon;
19 }
20
21 public void CommonMethod() {
22 System.out.println(getClass().getName());
23 }
24
25 }
但这里会有问题,啥问题呢?单线程中是没问题,但是多线程会出现问题,假设两个多线程A与B,A与B同时判断singleTon不为空,那会创建两次,那对于这个问题需要使用同步锁去解决了,下面改造下代码。
public class singleTonEx01 {
private static singleTonEx01 singleTon;
public static void main(String args[]) {
singleTonEx01 test1 = singleTonEx01.getInstance();
test1.CommonMethod();
}
// 防止使用new构造函数实例化对象
private singleTonEx01() {
}
public static singleTonEx01 getInstance() {
synchronized (singleTonEx01.class) {
if (null == singleTon)
singleTon = new singleTonEx01();
}
return singleTon;
}
public void CommonMethod() {
System.out.println(getClass().getName());
}
}
这还不是最好的方法,如果是这样的话每个线程都会使用同步锁代码块,使用同步锁其实是消耗资源的,因此,可以再改进下,在外面锁再加个判断,这样一来提高了效率。(推荐这种写法1)
另外,补充一下,实例变量加上volatile的意义。
创建对象可以分解为如下的3行伪代码memory=allocate(); //1:分配对象的内存空间ctorInstance(memory); //2:初始化对象instance=memory; //3:设置instance指向刚分配的内存地址上面3行代码中的2和3之间,可能会被重排序导致先3后2;
public class singleTonEx01 {
private volatile static singleTonEx01 singleTon;
public static void main(String args[]) {
singleTonEx01 test1 = singleTonEx01.getInstance();
test1.CommonMethod();
}
// 防止使用new构造函数实例化对象
private singleTonEx01() {
}
public static singleTonEx01 getInstance() {
if (singleTon == null) {
synchronized (singleTonEx01.class) {
if (null == singleTon)
singleTon = new singleTonEx01();
}
}
return singleTon;
}
public void CommonMethod() {
System.out.println(getClass().getName());
}
}
其实还有另外两种方法,一种是不管有没有调用都实例化(俗称恶汉式),另一种是静态内部类的方法(推荐这种写法2),线程安全而且高效
public class singleTonEx02 {
private static final singleTonEx02 singleTon=new singleTonEx02();
public static void main(String args[]) {
singleTonEx02 test1 = singleTon;
test1.CommonMethod();
}
// 防止使用new构造函数实例化对象
private singleTonEx02() {
}
public static singleTonEx02 getIntance() {
return singleTon;
}
public void CommonMethod() {
System.out.println(getClass().getName());
}
}
public class singleTonEx03 {
private static class SingleNBClass {
private final static singleTonEx03 singleTonEx03=new singleTonEx03();
}
public static void main(String args[]) {
singleTonEx03 test1 = singleTonEx03.getIntance();
test1.CommonMethod();
}
// 防止使用new构造函数实例化对象
private singleTonEx03() {
}
public static final singleTonEx03 getIntance() {
return SingleNBClass.singleTonEx03;
}
public void CommonMethod() {
System.out.println(getClass().getName());
}
}