这篇文章主要介绍“java设计模式中的单例模式简单介绍”,在日常操作中,相信很多人在java设计模式中的单例模式简单介绍问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java设计模式中的单例模式简单介绍”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!
一、引言
单例模式是啥?故名思意,就是单着的意思。没错,就是为了来保证整个系统运行中,从头至尾只有一个对象。比如说,我们最可爱的学校,可以有很多学生,可以有很多主任,但是不能有很多校长。为什么?因为要确保只有一个校长,学校这个系统才不会因为受干扰崩溃,所以单例模式应运而生。
二、实现方式
都知道了单例模式是干嘛的了,那就好办了。首先你要确保整个系统的laowang类只有老王一个对象 最重要的前提你要做什么??可想而知,老王不能被其他类所创造出来啊。
因此有如下做法:
1. 先把构造方法给私有化了(private)。
2. 接着在程序运行的时候创建一个对象放在内存里就得了。
你没看错,要实现单例模式,确确实实就只有这两步,第一步2秒钟搞定,第二步就是我们要来探讨的部分了。
实现单例模式有五种做法:
饿汉式:
也就是在程序装载时提前把对象创建了,有人来就给他。
懒汉式:
在有人需要的时候,再创建第一个对象,然后再给他。(懒加载)
双重检验方式:
内部类方式:
枚举方式:
提示:在上面实现方式中只展现线程安全的做法,详细的我后面会指出。
三、具体实现
1. 饿汉式
分为两步走:
把构造方法私有化
在程序装载时提前创建好实例
class Laowang{
private Laowang(){}//私有化构造方法
private static Laowang laowang=new Laowang();//直接创建静态实例
//对外提供静态方法获取当前的Laowang
public static Laowang getLaowang(){
return laowang;
}
}
//Main方法
public static void main(String[] args) {
//用Laowang类的静态方法getLaowang()获取实例;
Laowang laowang1=Laowang.getLaowang();
Laowang laowang2=Laowang.getLaowang();
//判断laowang1是否和老王2是同一个对象(是输出true,否则false)
System.out.println(laowang1==laowang2);
}
运行结果: true
上面这个例子中,在老王这个类中,先私有化构造方法,接着创建一个静态属性laowang, 然后提供一个对外的静态方法getLaowang()可以给别人拿这个laowang。(因为你已经把构造方法私有化了,所以你只能通过静态方法把laowang给别人。)
Main方法中定义了2个引用laowang1,laowang2,但是都是通过同一种方式拿到实例对象laowang.因此拿到的是同一个对象,所以返回true,这就是饿汉式实现法。
优点:实现简单,线程安全。
缺点:很明显,在类装载的时候直接创建,有时候你不需要它,它也会创 建,造成内存资源浪费。
饿汉式也有另外一种写法,也是一样的效果。把new Laowang()放在静态代码块里,如下:
class Laowang{
private static Laowang laowang;
private Laowang(){}//私有化构造方法
static {
laowang=new Laowang();
}
//对外提供静态方法获取当前的Laowang
public static Laowang getLaowang(){
return laowang;
}
}
2. 懒汉式
在程序需要用到调用的时候才给它(懒加载),因此做法如下:
class Laowang{
private static Laowang laowang;
private Laowang(){}//私有化构造方法
//对外提供静态方法,创建实例然后返回,当前的Laowang
public static synchronized Laowang getLaowang(){
if (laowang == null) {
laowang=new Laowang();
}
return laowang;
}
}
此做法需要在方法声明加上synchronized,(具体作用:比如说很多人来访问这个方法,他们必须排队访问) 这种怎么理解呢?就是说在别人需要用到laowang,调用getLaowang()的时候,先排队,排到他的时候,进去判断laowang是不是为空,是就new一个,不是就拿当前laowang给他。当然main运行结果还是为true这里就不作多的描述。
优点: 不会造成内存浪费
缺点: 很明显,人人平等,大家都要排队,既然排队就慢,高并发情况下,极度影响效率。
在这里解释为什么要加同步锁:如果不加的话,举个例子,程序运行刚开始,小黑和小红同时访问这个方法,同时作判断,肯定同时都判断为空,而且两个人都进去了,new Laowang();很明显直接造成laowang不是单例的了。因此要加锁。 小黑小红都要排队。
3. 双重检验锁
也是属于懒加载
class Laowang{郑州哪个人流医院好 http://www.csyhjlyy.com/
private volatile static Laowang laowang;//必须加上volatile 关键字
private Laowang(){}//私有化构造方法
//对外提供静态方法,创建实例然后返回,当前的Laowang
public static Laowang getLaowang(){
if (laowang == null) {
synchronized (Laowang.class){ //同步代码(照样要排队)
if (laowang==null){
laowang=new Laowang();
}
}
}
return laowang;
}
}
分析一下代码哈,首先静态属性laowang要加上volatile (具体作用要详细了解的话建议百度搜一下哈,属于多线程内容的一部分)。然后再getLaowang()方法中,先判断laowang是否为空,如果为空,请排队。排完队后,再次判断,如果还是为空,才new一个返回。
举个例子解释一下为什么要这样做:
还是小黑小红,同时并发进来访问,然后肯定同时第一次判断都为空,接着两个人排队,小黑先进去玩会,肯定第二次判断为空,结果肯定是小黑new了一个laowang走了。排到小红了,小红进来第二次判断发现laowang不为空了,直接带走。
温馨提示:再看一边再继续看下面内容
这个时候有人问了,那为什么要第一次判断干嘛,直接排队他不香吗?没错,我第一次也这不理解的地方。我们脑回路回退到小黑刚new完laowang走了。刚要排到小红了。突然来了个第三者小三,如果你没有第一次判断,小三还要继续排在小红后面,造成效率降低。但是现在小三第一次判断发现laowang已经不为空了(此时laowang是第一个人小黑弄出来的),直接带走。
优点: 解决了排队效率降低的问题,线程安全。
缺点: 实现较为复杂。
4. 內部类方式
也是属于懒加载,故名思意,首先整个内部类出来,代码如下:
兄台要看如下代码,请先了解final关键字的作用
这里对final作简单描述:
对类使用:表示该类不能被继承(俗称断子绝孙类)
对方法使用:表示该方法不能被重写
对基础类型使用:比如说int,float…表示该值不可以被更改
对引用对象使用:表示该引用从头到尾只指向一个对象
以上3.对基础类型使用,4.对引用对象使用都必须直接赋值。
class Laowang{
private Laowang(){} //私有化构造方法
//对外提供静态方法,调用内部类的属性,返回
public static final Laowang getLaowang(){
return laowangHolder.INSTANCE;
}
//静态内部类
private static final class LaowangHolder{
private static final Laowang INSTANCE =new Laowang();
}
}
解释以上代码:首先声明了一个内部类(LaowangHolder),他有个静态且被final修饰的属性INSTANCE,因此需要直接赋值,new Laowang();接着在getLaowang()方法中调用内部类的INSTANCE属性,返回。因为INSTANCE被final修饰,只指向同一个laowang,所以他是单例的。
5. 枚举方式
这些方法实现相对简单,所以直接上代码:
enum Laowang{
laowang;
public void whateverMethod(){}
}
//Main方法:
public static void main(String[] args) {
//直接当成属性调用就可以了
Laowang laowang1=Laowang.laowang;
Laowang laowang2=Laowang.laowang;
System.out.println(laowang1==laowang2);
}
直接声明一个枚举类,定义一个属性,main方法中直接获取即可。
到此,关于“java设计模式中的单例模式简单介绍”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注亿速云网站,小编会继续努力为大家带来更多实用的文章!
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。