1、什么是单例模式?
单例模式是一种对象创建模式,使用单例模式,可以保证为一个类只生成唯一的实例对象。
也就是说,在整个程序空间,该类只存在一个对象。
Gof对单例模式的定义是:保证一个类,只有一个实例存在,同时提供能对该实例加以访问的全局方法。
2、为什么要使用单例模式?
在应用系统开发中,我们常常有以下要求:
1、在多个线程之间,比如servlet环境,共享同一个资源或者操作同一个对象
2、在整个程序空间使用全局变量,共享资源
3、大规模系统中,为了性能的考虑,需要节省对象创建时间等等。
因为Sinleton模式可以保证为一个类只生成唯一的实例对象,所以这些,Singleton模式就派上用场了。
3、单例模式的创建方式
1、懒汉式
2、饿汉式
3、双重检查模式’
4、代码实现分析
1、普通对象创建
package com.sinwao.singleton;
public class Person {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
2、懒汉式创建单例
package com.sinwao.singleton;
public class LazyPerson {
public static final LazyPerson hp = new LazyPerson();
private LazyPerson() {}
public static LazyPerson getInstance() {
return hp;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
该模式线程始终是安全,但是比较消耗内存资源
3、饿汉式创建单例
package com.sinwao.singleton;
public class HungryPerson {
public static HungryPerson lp = null;
private HungryPerson() {}
public static HungryPerson getInstance() {
if (lp == null) {
lp = new HungryPerson();
}
return lp;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
该模式在多线程访问同一对象时,可能会出现重复创建对象,于是又了双重检查机制
4、双重检查创建单例
package com.sinwao.singleton;
public class DoubleCheckPerson {
public static DoubleCheckPerson dcp = null;
private DoubleCheckPerson() {}
public static DoubleCheckPerson getInstance() {
if (dcp == null) {
synchronized (DoubleCheckPerson.class) {
if (dcp == null) {
dcp = new DoubleCheckPerson();
}
}
}
return dcp;
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
该模式线程是安全的,在多线程情况下也不会出现重复创建对象的情况,效率较高,内存开销也较低。注意一点:面试的时候我也被问到过这种情况
5、测试
package com.sinwao.singleton;
/**
* 单例模式
* @author Administrator
*
*/
public class TestSingleton {
public static void main(String[] args) {
System.out.println("一般对象创建");
Person p1 = new Person();
Person p2 = new Person();
p1.setName("T-Mac");
p2.setName("Kobe");
System.out.println(p1.getName());
System.out.println(p2.getName());
********一般对象创建,一个对象对应一个实例************///
System.out.println("懒汉式单例创建");
LazyPerson hp1 = LazyPerson.getInstance();
LazyPerson hp2 = LazyPerson.getInstance();
hp1.setName("T-Mac");
hp2.setName("Kobe");
System.out.println(hp1.getName());
System.out.println(hp2.getName());
***********懒汉式单例创建*****************************
System.out.println("饿汉式单例创建");
HungryPerson lp1 = HungryPerson.getInstance();
HungryPerson lp2 = HungryPerson.getInstance();
lp1.setName("T-Mac");
lp2.setName("Kobe");
System.out.println(lp1.getName());
System.out.println(lp2.getName());
***********饿汉式单例创建*****************************
System.out.println("双重检查单例创建");
DoubleCheckPerson dcp1 = DoubleCheckPerson.getInstance();
DoubleCheckPerson dcp2 = DoubleCheckPerson.getInstance();
dcp1.setName("T-Mac");
dcp2.setName("Kobe");
System.out.println(dcp1.getName());
System.out.println(dcp2.getName());
***********双重检查单例创建*****************************
}
}