单例模式特点:
1.单例模式只能有一个实例
2.单例类必须自己创建自己的唯一实例
3,单例类对外部其他对象必须提供同一实例
例如:
饿汉式单例模式
定义一个单例类SingleEHanModel (饿汉式),饿汉式:类加载的过程中,会将单例对象仅且一次实例化
public class SingleEHanModel {
//TODO 1.整个应用程序只能有1个实例
private static SingleEHanModel instance = new SingleEHanModel();
//TODO 2.必须自己创建自己实例(私有化构造器)
private SingleEHanModel(){}
//TODO 3.提供外部静态获取该实例的方法
public static SingleEHanModel getInstance(){
return instance;
}
}
外部类SingleTestMain获取单例实例
public class SingleTestMain {
public static void main(String[] args) {
//获取单例对象
SingleEHanModel instance = SingleEHanModel.getInstance();
System.out.println(instance);
}
}
输出单例对象
饿汉单例模式优点:
1.由于一开始已经实例化对象,在多线程情况下不会出现线程问题(线程安全)
2.没有使用锁实现线程同步,执行效率较高
饿汉单例模式缺点:
1.由于一开始类加载会实例化对象,会消耗内存
2.若单例对象没有被使用,被当成垃圾对象回收
懒汉式单例模式
定义一个单例类SingleLanHanModel (饿汉式),懒汉式:顾名思义这个类很懒,等需要用到这个类才会去实例化单例对象
public class SingleLanHanModel {
//TODO 1.整个应用程序只能有1个实例
private static SingleLanHanModel instance;
//TODO 2.必须自己创建自己实例(私有化构造器)
private SingleLanHanModel(){}
//TODO 3.提供外部静态获取该实例的方法
public static SingleLanHanModel getInstance(){
if (null == instance){
instance = new SingleLanHanModel();
}
return instance;
}
}
外部类SingleTestMain获取单例实例
public static void main(String[] args) {
//获取单例对象
SingleLanHanModel instance = SingleLanHanModel.getInstance();
System.out.println(instance);
}
输出单例对象
懒汉单例模式优点:
1.需要用到才会实例化对象,类似@Lazy懒加载
懒汉单例模式缺点:
1.由于需要用到这个类才会实例化,在多线程情况下容易出现线程问题,实例化的单例对象不是同一个
多线程情况下,容易出现实例化对象不是同一个 如图
public static void main(String[] args) {
for (int i = 0; i < 3; i++) {
Thread thread = new Thread(() -> {
System.out.println(SingleLanHanModel.getInstance());
});
thread.start();
}
}
解决方法:私有化构造方法加锁:synchronized,当A、B线程同时调用实例方法getInstance(),A会先获取锁执行完释放锁,B线程才会继续执行
或者方法二
总结:
单例模式共同优点:
1、 避免对象频繁的创建和销毁
2、 避免对资源的多重占用,例如写入文件
单例模式应用场景:
在工作中用到单例模式的应用场景
例如:
1、 整个应用程序的日志对象,比如log4j实例化一次在整个应用程序使用该单例对象去日志输出
2、 自定义数据库连接池设计,例如自己封装的JDBC数据库连接池和事务
3、 自定义多线程的线程池设计,例如Redis的线程池设计
4、 网站的计数器,在redis没出之前统计点赞数量之前用到单例模式去计数