一、什么是单例模式
单例模式是一种对象创建型模式,使用单例模式,可以保证为一个类只生成唯一的实例对象。也就是说,在整个程序空间中,该类只存在一个实例对象。 其实,GoF对单例模式的定义是:保证一个类、只有一个实例存在,同时提供能对该实例加以访问的全局访问方法。
二、为什么要使用单例模式呢?
在应用系统开发中,我们常常有以下需求:
- 在多个线程之间,比如servlet环境,共享同一个资源或者操作同一个对象
- 在整个程序空间使用全局变量,共享资源
- 大规模系统中,为了性能的考虑,需要节省对象的创建时间等等。
因为Singleton模式可以保证为一个类只生成唯一的实例对象,所以这些情况,Singleton模式就派上用场了。
三、单例模式的实现及其原理解析
1.饿汉式。
代码:
public class Person {
public static final Person person = new Person();
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
//构造函数私有化
private Person() {
}
//提供一个全局的静态方法
public static Person getPerson() {
return person;
}
}
main方法
public class MainClass {
public static void main(String[] args) {
Person per = Person.getPerson();
Person per2 = Person.getPerson();
per.setName("zhangsan");
per2.setName("lisi");
System.out.println(per.getName());
System.out.println(per2.getName());
}
}
原理解析:此种方式无论是单线程或者多线程都可以实现单例功能。原因是使用final修饰以后,JVM是把这种类型的变量作为全局共享。所以一旦创建,所有的对象都可以查询到,且因为final 修饰不可更改。
2.懒汉式。
代码:
public class Person2 {
private String name;
private static Person2 person;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
//构造函数私有化
private Person2() {
}
//提供一个全局的静态方法
public static Person2 getPerson() {
if(person == null) {
person = new Person2();
}
return person;
}
}
main方法不变
解析:此种方法不能再多线程中使用,对象放到JVM的heap中,虽然heap是线程共享,但是如果线程数量过多,多个线程同时创建对象的话,就极有可能创建多个对象。此种问题也就涉及到线程的高并发问题。解决办法见下。
3.双重检查。
此种方法为了解决懒汉式创建单例线程不安全问题。
public class Person4 {
private String name;
private static Person4 person;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
//构造函数私有化
private Person4() {
}
//提供一个全局的静态方法
public static Person4 getPerson() {
if(person == null) {
synchronized (Person4.class) {
if(person == null) {
person = new Person4();
}
}
}
return person;
}
}
解析:通过给代码块加锁,保证只有一个线程new 对象。同时sychinize时尽量保证给需要锁的位置加锁,避免不必要的阻断,可以大大提高程序运行效率。