定义:
单例模式,是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中一个类只有一个实例。即一个类只有一个对象实例。
特点
- 单例类只能有一个实例。
- 单例类必须自己自己创建自己的唯一实例。
- 单例类必须给所有其他对象提供这一实例
单例模式构造要点
- 私有的构造方法
- 指向自己实例的私有静态引用
以自己实例为返回值的静态的共有方法
饿汉式
public class Singleton {
private static Singleton singleton = new Singleton();
private Singleton(){}
public static Singleton getInstance(){
return singleton;
}
}
懒汉式
public class Singleton {
private static Singleton singleton;
private Singleton(){}
//加锁保持同步
public static synchronized Singleton getInstance(){
if(singleton==null){
singleton = new Singleton();
}
return singleton;
}
}
上面的方式考虑到线程安全的问题,线程不安全时会出现多个Singleton实例,
双重锁的形式
public class Singleton{
private static volatile Singleton instance=null;
private Singleton(){
//TO DO
}
public static Singleton getInstance(){
if(instance==null){
synchronized(SingletonClass.class){
if(instance==null){
instance=new Singleton();
}
}
}
return instance;
}
}
这个模式将同步内容下方到if内部,提高了执行的效率,不必每次获取对象时都进行同步,只有第一次才同步,创建了以后就没必要了。
这种模式中双重判断加同步的方式,比第一个例子中的效率大大提升,因为如果单层if判断,在服务器允许的情况下,假设有一百个线程,耗费的时间为100*(同步判断时间+if判断时间),而如果双重if判断,100的线程可以同时if判断,理论消耗的时间只有一个if判断的时间。
所以如果面对高并发的情况,而且采用的是懒汉模式,最好的选择就是双重判断加同步的方式。
单例模式的优缺点
优点:
1.在内存中只有一个对象,节省内存空间。
2. 避免频繁的创建销毁对象,可以提高性能。
3.避免对共享资源的多重占用。
4.可以全局访问。
缺点:
1.扩展困难,由于getInstance静态函数没有办法生成子类的实例。如果要拓展,只有重写那个类
2. 隐式使用引起类结构不清晰。
3. 导致程序内存泄露的问题。
适用场景:
由于单例模式的以上优点,所以是编程中用的比较多的一种设计模式。以下为使用单例模式的场景:
1,需要频繁实例化然后销毁的对象。
2,创建对象时耗时过多或者耗资源过多,但又经常用到的对象。
3,资源共享的情况下,避免由于资源操作时导致的性能或损耗等
4,控制资源的情况下,方便资源之间的互相通信。
单例模式注意事项:
只能使用单例类提供的方法得到单例对象,不要使用反射,否则将会实例化一个新对象。
不要做断开单例类对象与类中静态引用的危险操作。
多线程使用单例使用共享资源时,注意线程安全问题。
Java中单例模式的一些常见问题
1.单例模式的对象长时间不用会被jvm垃圾收集器收集吗
人为的断开静态引用到单例对象的连接,否则jvm虚拟机不会回收单例对象
jvm回收类的判定条件:
1.该类的所有实例都已经回收,也就是java堆中不存在该类的任何实例
2.加载该类的classLoader已经回收
3.该类对应的java.lang.Class对象没有任何地方被引用,无法在任何地方通过反射访问该类的方法
满足上述三个条件jvm才会垃圾收集是回收类
2.在jvm中会出现多个单例对象吗?
在同一个jvm中使用单例模式中只能得到同一个单例,除非使用反射将可得到新的实例
try {
Class c=Class.forName(Singleton.class.getName());
try {
Constructor ct=c.getDeclaredConstructor();
ct.setAccessible(true);
Singleton singleton=(Singleton)ct.newInstance();
toastShort("singleton的实例通过反射获取的");
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
} catch (ClassNotFoundException e) {
e.printStackTrace();
}