一、什么是单例模式?
单例模式是一种对象创建型模式,使用单例模式,可以保证为一个类只生成唯一的实例对象。也就是说,在整个程序空间中,该类只存在一个实例对象。其实,GoF对单例模式的定义是:保证一个类、只有一个实例存在,同时提供能对该实例加以访问的全局访问方法。
二、为什么要使用单例模式呢?
在应用系统开发中,我们常常有以下需求:
(1)、在多个线程之间,比如servlet环境,共享同一个资源或者操作同一个对象
(2)、在整个程序空间使用全局变量,共享资源
(3)、大规模系统中,为了性能的考虑,需要节省对象的创建时间等等。
因为SingIeton模式可以保证为一个类只生成唯一的实例对象,所以这些情况,SingIeton模式就派上用场了。
三、单例模式的实现
(1)、饿汉式
(2)、懒汉式
(3)、双重检查
四、代码实现:
(1)、饿汉模式:
创建Person类
package com.renxin.singleton;
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;
}
}
创建测试类MainClass
package com.renxin.singleton;
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(per.getName());
}
}
控制台打印结果为:
lisi
lisi
(2)、懒汉式
创建Persion2类:
package com.renxin.singleton;
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;
}
}
创建测试类MainClass:
package com.renxin.singleton;
public class MainClass {
public static void main(String[] args) {
Person2 per = Person2.getPerson();
Person2 per2 = Person2.getPerson();
per.setName("zhangsan");
per2.setName("lisi");
System.out.println(per.getName());
System.out.println(per.getName());
}
}
控制台打印结果为:
lisi
lisi
五、懒汉式和饿汉式比较:
饿汉式是线程安全的,在类创建的同时就已经创建好一个静态的对象供系统使用,以后不在改变。
懒汉式如果在创建实例对象时不加上synchronized则会导致对对象的访问不是线程安全的。
推荐使用饿汉式:
从实现方式来讲他们最大的区别就是懒汉式是延时加载,他是在需要的时候才创建对象,而饿汉式在虚拟机启动的时候就会创建,饿汉式无需关注多线程问题、写法简单明了、能用则用。但是它是加载类时创建实例、所以如果是一个工厂模式、缓存了很多实例、那么就得考虑效率问题,因为这个类一加载则把所有实例不管用不用一块创建。
懒汉式的优点是延时加载、缺点是应该用同步(想改进的话现在还是不可能,比如double-check)、其实也可以不用同步、看你的需求了,多创建一两个无引用的废对象其实也没什么大不了。
(3)、双重检查:
package com.renxin.singleton;
public class Person3 {
private String name;
private static Person3 person;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
//构造函数私有化
private Person3() {
}
//提供一个全局的静态方法
public static Person3 getPerson() {
if(person == null) {
synchronized (Person3.class) {
if(person == null) {
person = new Person3();
}
}
}
return person;
}
}
双重检查说白了就是对懒汉式的改进!