场景问题
读取配置文件的内容,这些配置文件都是由开发人员定义的,在里面定义一些应用需要的参数,当然在实际应用中这种配置文件的格式有不一样,springboot有yml,properties格式的,ssm有xml格式的
不同模式的解决方案
使用单例模式解决问题
直接使用java读取配置,这里不做代码说明,但是我们可以想像,客户端要想使用我们的配置类,需要new 一个配置App的实例,如果我们很多地方都需要用到这个配置文件,那么客户端必须new 很多个这样的实例,这就造成了资源的浪费,事实上,我们只需要一个配置类文件,这时候我们可以使用单例模式来解决上述问题
定义
保证一个类仅有一个实例,并提供他的全局访问点,把创建实例的权限收回,让类自身来负责自己类实例的创建工作,然后由这个类提供外部可以访问这个类实例的方法
示例代码
其实在java中,单例模式的实现分为俩种。一种为懒汉式,一种为饿汉式
我们先看懒汉式
/**
* 懒汉模式的单例实现
* @author 谭婧杰
* @data 2020/1/17
*/
public class Singleton {
private static Singleton unquesingleton;
public Singleton() {
}
public static synchronized Singleton getInstance(){
if (unquesingleton == null){
unquesingleton = new Singleton();
}
return unquesingleton;
}
}
饿汉式
/** 饿汉式单例模式实现
* @author 谭婧杰
* @data 2020/1/17
*/
public class Singletons {
private static Singleton uniqueInstance = new Singleton();
public static Singleton getInstance(){
return uniqueInstance;
}
}
对比俩段代码我们可以发现,这两种模式的区别,即类创建的时机,一种是在类加载就开始创建,一种是等到需要时候在进行创建
模式详解
单例模式主要保证这个类在运行期间只会被创建一个实例,另外单例模式还提供了一个全局访问点,就是getInstance方法,他只关心类的创建问题,不关心类的实现
范围
Java 中实现的单例是一个虚拟机的范围,因为装载类的功能是虚拟机的,所以一个虚拟机在通过自己的ClassLoader时会创建一个实例
实现
懒汉式
- 私有化构造方法
- 提供获取实例的方法
- 将获取实例的方法变为静态的
- 定义存储实例的属性将这个属性也设置为静态的
饿汉式
这种模式和上面一种模式区别只在于怎么实现getInstance这个方法,如果我们要控制只创建一个实例,我们可以知道java中static的特性,他只在类装载的时候进行初始化,多个实例的static会共享一块变量
懒汉式是线程不安全的,但是我们可以使用双重锁技制来实现他的线程安全
/**
* 懒汉模式的单例实现
* 双重锁机制
* @author 谭婧杰
* @data 2020/1/17
*/
public class Singleton {
private static volatile Singleton unquesingleton;
public Singleton() {
}
public static synchronized Singleton getInstance(){
if (unquesingleton == null){
synchronized (Singleton.class){
if(unquesingleton == null){
unquesingleton = new Singleton();
}
}
}
return unquesingleton;
}
}
扩展
java中更好单例模式的实现方式
类级同步类
有static修饰的成员式内部类,如果没有则被称之为对象级内部类
多线程默认同步锁
多线程开发中为了解决并发问题,主要通过同步锁来进行同步控制,但是在一些情况下,jvm已经为您隐含的执行了同步,如
- 由静态初始化器初始化数据时
- 访问final字段时
- 在创建线程之前创建对象时候
- 线程可以看见它将要处理的对象时
即我们可以使用静态初始化器,由JVM 来保证线程安全
本质
控制实例数目,控制在运行期间,一些类的实例可能只有一个,如果是多个的话,我们也可以采用MAP 缓存来缓存实现单例的示例