您的位置:首页 > 编程语言 > Java开发

Java单例模式

2016-09-28 12:30 204 查看
简单介绍

在刚开始接触Java的构造方法的时候,看到构造方法用的都是public,突然想如果用private修饰会怎样,试了一下,创建对象的时候报错,例子如下图:




上网查了查,发现的确有这种private的构造方法,叫做单例模式。构造函数弄成private 就是单例模式,即不想让别人用new 方法来创建多个对象,可以在类里面先生成一个对象,然后写一个public static方法把这个对象return出去。


单例模式有以下特点:

  1、单例类只能有一个实例。

  2、单例类必须自己创建自己的唯一实例。

  3、单例类必须给所有其他对象提供这一实例。

  

几种常见的单例模式

饿汉式

//静态方法只在编译期间执行一次初始化,以后不再改变,也就是只会有一个对象。线程安全。
public class Singleton1 {
private Singleton1() {}
private static final Singleton1 single = new Singleton1();

public static Singleton1 getInstance() {
return single;
}
}


懒汉式

一次判断

//那个if判断确保对象只创建一次,并加上synchronized保证线程安全
public class Singleton2 {
private Singleton2() {}
private static Singleton2 single = null;

public static synchronized Singleton2 getInstance() {
if (single == null) {
single = new Singleton();
}
return single;

}


两次判断

//在getInstance中做了两次null检查,确保了只有第一次调用单例的时候才会做同步,这样也是线程安全的,同时避免了每次都同步的性能损耗
public static Singleton2 getInstance() {
if (single == null) {
synchronized (Singleton2.class) {
if (single == null) {
singleton = new Singleton2();
}
}
}
return singleton;
}


登记式单例类

//登记式单例实际上维护了一组单例类的实例,将这些实例存放在一个Map(登记薄)中,对于已经登记过的实例,则从Map直接返回,对于没有登记的,则先登记,然后返回。
//类似Spring里面的方法,将类名注册,下次从里面直接获取。
public class Singleton3 {
private static Map<String,Singleton3> map = new HashMap<String,Singleton3>();
static{
Singleton3 single = new Singleton3();
map.put(single.getClass().getName(), single);
}
//保护的默认构造子
protected Singleton3(){}
8d02
//静态工厂方法,返还此类惟一的实例
public static Singleton3 getInstance(String name) {
if(name == null) {
name = Singleton3.class.getName();
System.out.println("name == null"+"--->name="+name);
}
if(map.get(name) == null) {
try {
map.put(name, (Singleton3) Class.forName(name).newInstance());
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
return map.get(name);
}
//一个示意性的商业方法
public String about() {
return "Hello, I am RegSingleton.";
}
public static void main(String[] args) {
Singleton3 single3 = Singleton3.getInstance(null);
System.out.println(single3.about());
}
}


小总结

饿汉式在类创建的同时就实例化一个静态对象出来,不管之后会不会使用这个单例,都会占据一定的内存,但是相应的,在第一次调用时速度也会更快,因为其资源已经初始化完成,

而懒汉式顾名思义,会延迟加载,在第一次使用该单例的时候才会实例化对象出来,第一次调用时要做初始化,如果要做的工作比较多,性能上会有些延迟,之后就和饿汉式一样了。

参考文献:http://blog.csdn.net/jason0539/article/details/23297037/
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: