给女友讲讲设计模式——单例模式(JAVA实例)2
前言
曾经有两个人,一个是A,一个是B,他们俩都是这个世界上独立的个体,每个人都是这个世界上独一无二的。这个A呢,在一出生的时候,别人就会把他需要的食物全部给他,然后等到他饿了的时候,就可以直接拿过来吃,但是在他不饿的时候,天天被在身上确是一种负担,人们都管他叫做饿汉。B呢,则是在出生的时候,并没有任何食物给他,但是在他饿的时候,他连伸手都不用,只需要张嘴喊一嗓子,便有人给他送来了食物,就因为他这么懒,所以被叫做懒汉。为什么要讲这个故事呢,接下来你就会明白了。
单例模式
在web开发中,我们经常碰到这样一种情况,我们在整个项目的上下文中,有且只有一个实例,所有线程操纵的都是它,例如说Mybatis中的SqlSessionFactory。如果说只存在一个实例,那么他绝对是不可以new出来,所以他的构造方法一定是私有的。
单例模式从方式上分为懒汉模式,和饿汉模式,这就像是咱们刚刚提到故事里的B和A一样,具体代码如下:
package singleton;
/**
* 这是懒汉模式,但是对于多线程的情况下是不安全的
*
* @author luckyharry
*
*/
public class SingletonLazyNotSafe {
private SingletonLazyNotSafe() {
System.out.println("单例懒汉模式,无锁,实例化--");
}
private static SingletonLazyNotSafe singletonLazyNotSafe;
public static SingletonLazyNotSafe getInstance() {
if (singletonLazyNotSafe == null) {
singletonLazyNotSafe = new SingletonLazyNotSafe();
}
return singletonLazyNotSafe;
}
}
这是懒汉模式,但是我们会发现getInstance()方法,多个线程可能会共同竞争,将会导致数据的不统一性,那么我们需要给它添加一个锁,关键字synchronized。
package singleton;
/**
* 这是懒汉模式,对于多线程的情况下是安全的 但是因为加了锁,所以效率比较低
*
* @author luckyharry
*
*/
public class SingletonLazySafe {
private SingletonLazySafe() {
System.out.println("懒汉模式 有锁 实例化--");
}
private static SingletonLazySafe singletonLazySafe;
public static synchronized SingletonLazySafe getInstance() {
if (singletonLazySafe == null) {
singletonLazySafe = new SingletonLazySafe();
}
return singletonLazySafe;
}
}
懒汉模式,他是在当需要获得的时候才实例化,就像是刚刚所说的B那个人一样,实现了懒加载。但是加了锁之后,效率比较低下。
package singleton;
public class SingletonHunger {
private static SingletonHunger singletonHunger=new SingletonHunger();
// 构造器为私有的,不可以通过new实力化
private SingletonHunger() {
System.out.println("初始化 Singleton 饿汉方式--");
}
public static SingletonHunger getInstance() {
return singletonHunger;
}
}
饿汉模式我们不再去考虑多线程的问题,但是它却是在类加载的时候便会实例化单例本身,并未实现懒加载。这就像是刚刚说的A一样,在一出生的时候,该实例化的已经被实例化放在那了。
package singleton;
public class SingletonInnerClass {
// 构造器为私有的,不可以通过new实力化
private SingletonInnerClass() {
System.out.println("初始化 Singleton 内部类方式--");
}
private static class SingletonCreate {
private static SingletonInnerClass singleton = new SingletonInnerClass();
}
public static SingletonInnerClass getInstance() {
return SingletonCreate.singleton;
}
}
这种内部类的形式,既解决了锁的问题,又实现了懒加载,是一种懒汉模式的变种。
接下来就是测试类了。
package singleton;
/**
* 懒汉模式以及饿汉模式的本质区别在于
* 单例的类的初始化的位置,如果我们是在类刚开始初始化的时候就初始化了的化,那么是饿汉加载
* 如果我们是在获取实例的那个方法初始化的化(实现了懒加载),那就是懒汉模式。
* @author luckyharry
*
*/
public class MainTest {
public static void main(String[] args) {
//获得单例中的实例
SingletonInnerClass.getInstance();
SingletonLazyNotSafe.getInstance();
SingletonLazySafe.getInstance();
SingletonHunger.getInstance();
}
}
后记
我会一直以这样通俗易懂的自己能想到的例子,讲解设计模式,最大的启发是来源于《大话设计模式》。非常感觉这本书的作者程杰先生。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
-
上一篇
说说Javac
Java语言有Java语言的规范,,这个规范详细描述了Java语言有哪些词法、语法,而Java虚拟机也有其Java虚拟机的规范,同样Java虚拟机的规范和Java语言规范并不一样,它们都有自己的词法和语法解析规则,而且解析规则也是不同的。那么如何将Java的语法规则适应到Java虚拟机的语法规则呢?这个任务就由Javac编译器来完成。它的任务就是将Java语言规范转化成Java虚拟机语言规范,完成"翻译工作"。 本文主要围绕如下三个方面来讲解: 第一、Javac是什么; 第二、Javac编译器的基本结构; 一、Javac是什么 Javac是一种编译器,能将一种语言规范转成另外一种语言规范,通常编译器都是将人易理解的语言规范转化成机器易理解的语言规范,如C、C++、Python或者汇编语言都是将源码直接编译成目标机器码,这个目标机器码是CPU直接执行的指令集合。这些指令集合也就是底层的一种语言规范,这种语言规范机器能够直接识别,但是人不可能直接去写目标机器码。虽然这种机器码执行非常高效,但是对人太不友好了,往往开发这个代码的成本远远高于省下的机器的执行成本,所以才有了编译器,有了编...
-
下一篇
BAT面试必备——Java 集合类
本文首发于我的个人博客:尾尾部落 来源:http://www.runoob.com/java/java-collections.html 来源:https://www.cnblogs.com/jing99/p/7057245.html 1. Iterator接口 Iterator接口,这是一个用于遍历集合中元素的接口,主要包含hashNext(),next(),remove()三种方法。它的一个子接口LinkedIterator在它的基础上又添加了三种方法,分别是add(),previous(),hasPrevious()。也就是说如果是先Iterator接口,那么在遍历集合中元素的时候,只能往后遍历,被遍历后的元素不会在遍历到,通常无序集合实现的都是这个接口,比如HashSet,HashMap;而那些元素有序的集合,实现的一般都是LinkedIterator接口,实现这个接口的集合可以双向遍历,既可以通过next()访问下一个元素,又可以通过previous()访问前一个元素,比如ArrayList。 2. List List是元素有序并且可以重复的集合。 List的主要实现:Arr...
相关文章
文章评论
共有0条评论来说两句吧...