Java设计模式之(一)------单例模式
1、什么是单例模式?
采取一定的办法保证在整个软件系统中,单例模式确保对于某个类只能存在一个实例。有如下三个特点:
①、单例类只能有一个实例
②、单例类必须自己创建自己的实例
③、单例类必须提供外界获取这个实例的方法
2、单例类的设计思想(Singleton)
①、外界不能创建这个类的实例,那么必须将构造器私有化。
public class Singleton {
//构造器私有化
private Singleton(){
}
}
②、单例类必须自己创建自己的实例,不能允许在类的外部修改内部创建的实例,所以将这个实例用 private 声明。为了外界能访问到这个实例,我们还必须提供 get 方法得到这个实例。因为外界不能 new 这个类,所以我们必须用 static 来修饰字段和方法。
//在类的内部自己创建实例
private static Singleton singleton = new Singleton(); //提供get 方法以供外界获取单例
public Singleton getInstance(){
return singleton;
}
3、单例模式之饿汉模式
public class Singleton {
//构造器私有化
private Singleton(){
}
//在类的内部自己创建实例
private static Singleton singleton = new Singleton();
//提供get 方法以供外界获取单例
public static Singleton getInstance(){
return singleton;
}
}
测试:
public static void main(String[] args) {
Singleton s1 = Singleton.getInstance();
Singleton s2 = Singleton.getInstance();
System.out.println(s1.equals(s2)); //true
}
这种模式避免了多线程的同步问题,不过在 类装载的时候就进行了实例化,有可能这个实例化过程很长,那么就会加大类装载的时间;有可能这个实例现阶段根本用不到,那么创建了这个实例,也会浪费内存。没有达到 lazy-loading 的效果。
4、单例模式之懒汉模式(线程不安全)
//懒汉模式
public class Singleton {
//构造器私有化
private Singleton(){ }
//在类的内部自己创建实例的引用
private static Singleton singleton = null; //提供get 方法以供外界获取单例
public static Singleton getInstance(){
if(singleton == null){
singleton = new Singleton();
}
return singleton;
} }
这种方法达到了 lazy-loading 的效果,即我们在第一次需要得到这个单例的时候,才回去创建它的实例,以后再需要就可以不用创建,直接获取了。但是这种设计在多线程的情况下是不安全的。

我们可以创建两个线程来看看这种情况:
public class ThreadSingleton extends Thread{
@Override
public void run() {
try {
System.out.println(Singleton.getInstance());
} catch (Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
ThreadSingleton s1 = new ThreadSingleton();
s1.start(); //com.ys.pattern.Singleton@5994a1e9
ThreadSingleton s2 = new ThreadSingleton();
s2.start(); //com.ys.pattern.Singleton@40dea6bc
}
}
很明显:最后输出结果的两个实例是不同的。这便是线程安全问题。那么怎么解决这个问题呢?
参考这篇博客:Java多线程同步:http://www.cnblogs.com/ysocean/p/6883729.html
5、单例模式之懒汉模式(线程安全)
这里我们采用同步代码块来达到线程安全
//懒汉模式线程安全
public class Singleton {
//构造器私有化
private Singleton(){ }
//在类的内部自己创建实例的引用
private static Singleton singleton = null; //提供get 方法以供外界获取单例
public static Singleton getInstance() throws Exception{
synchronized (Singleton.class) {
if(singleton == null){
singleton = new Singleton();
}
}
return singleton;
} }
6、单例模式之懒汉模式(线程安全)--双重校验锁
分析:上面的例子我们可以看到,synchronized 其实将方法内部的所有语句都已经包括了,每一个进来的线程都要单独进入同步代码块,判断实例是否存在,这就造成了性能的浪费。那么我们可以想到,其实在第一次已经创建了实例的情况下,后面再获取实例的时候,可不可以不进入这个同步代码块?
//懒汉模式线程安全--双重锁校验
public class Singleton {
//构造器私有化
private Singleton(){ }
//在类的内部自己创建实例的引用
private static Singleton singleton = null; //提供get 方法以供外界获取单例
public static Singleton getInstance() throws Exception{
if(singleton == null){
synchronized (Singleton.class) {
if(singleton == null){
singleton = new Singleton();
}
}
}
return singleton;
} }
以上的真的完美解决了单例模式吗?其实并没有,请看下面:
7、单例模式之最终版
我们知道编译就是将源代码翻译成机械码的过程,而Java虚拟机的目标代码不是本地机器码,而是虚拟机代码。编译原理里面有个过程是编译优化,就是指在不改变原来语义的情况下,通过调整语句的顺序,来让程序运行的更快,这个过程称为 reorder。
JVM 只是一个标准,它并没有规定有关编译器优化的内容,也就是说,JVM可以自由的实现编译器优化。
那么我们来再来考虑一下,创建一个变量需要哪些步骤?
①、申请一块内存,调用构造方法进行初始化
②、分配一个指针指向该内存
而这两步谁先谁后呢?也就是存在这样一种情况:先开辟一块内存,然后分配一个指针指向该内存,最后调用构造方法进行初始化。
那么针对单例模式的设计,就会存在这样一个问题:线程 A 开始创建 Singleton 的实例,此时线程 B已经调用了 getInstance的()方法,首先判断 instance 是否为 null。而我们上面说的那种模型, A 已经把 instance 指向了那块内存,只是还没来得及调用构造方法进行初始化,因此 B 检测到 instance 不为 null,于是直接把 instance 返回了。那么问题出现了:尽管 instance 不为 null,但是 A 并没有构造完成,就像一套房子已经给了你钥匙,但是里面还没有装修,你并不能住进去。
解决方案:使用 volatile 关键字修饰 instance
我们知道在当前的Java内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进行读写。这就可能造成一个线程在主存中修改了一个变量的值,而另外一个线程还继续使用它在寄存器中的变量值的拷贝,造成数据的不一致。
volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。
//懒汉模式线程安全--volatile
public class Singleton {
//构造器私有化
private Singleton(){ }
//在类的内部自己创建实例的引用
private static volatile Singleton singleton = null; //提供get 方法以供外界获取单例
public static Singleton getInstance() throws Exception{
if(singleton == null){
synchronized (Singleton.class) {
if(singleton == null){
singleton = new Singleton();
}
}
}
return singleton;
} }
到此我们完美的解决了单例模式的问题。但是 volatile 关键字是 JDK1.5 才有的,也就是 JDK1.5 之前是不能这样用的
PS:我们还可以使用 枚举类型 或静态内部类来实现单例模式
8、单例模式之枚举类
public enum Singleton{
INSTANCE;
private Singleton(){}
}
9、单例模式之静态内部类
public class InnerSingleton {
private InnerSingleton(){}
public static InnerSingleton getInstance(){
return Inner.instance;
}
static class Inner{
static InnerSingleton instance = new InnerSingleton();
}
public static void main(String [] args){
System.out.println(InnerSingleton.getInstance()==InnerSingleton.getInstance());//true
System.out.println(InnerSingleton.getInstance().equals(InnerSingleton.getInstance()));//true
}
单例模式的应用:
1、windows 系统的回收站,我们能在任何盘符删除数据,但是最后都是到了回收站中
2、网站的计数器,不过不采用单例模式,很难实现同步
3、数据库连接池,可以节省打开或关闭数据库连接所引起的效率损耗,用单例模式来维护,可以大大降低这种损耗。
由上可以总结单例模式的应用场景:
①、资源共享
②、方便资源互相通信
Java设计模式之(一)------单例模式的更多相关文章
- Java设计模式之《单例模式》及应用场景
摘要: 原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6510196.html 所谓单例,指的就是单实例,有且仅有一个类实例,这个单例不应该 ...
- Java设计模式之【单例模式】
Java设计模式之[单例模式] 何为单例 在应用的生存周期中,一个类的实例有且仅有一个 当在一些业务中需要规定某个类的实例有且仅有一个时,就可以用单例模式 比如spring容器默认初始化的实例就是单例 ...
- Java设计模式中的单例模式
有时候在实际项目的开发中,我们会碰到这样一种情况,该类只允许存在一个实例化的对象,不允许存在一个以上的实例化对象,我们将这种情况称为Java设计模式中的单例模式.设计单例模式主要采用了Java的pri ...
- 重学 Java 设计模式:实战单例模式
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 5个创建型模式的最后一个 在设计模式中按照不同的处理方式共包含三大类:创建型模式.结 ...
- Java设计模式4:单例模式
前言 非常重要,单例模式是各个Java项目中必不可少的一种设计模式.本文的关注点将重点放在单例模式的写法以及每种写法的线程安全性上.所谓"线程安全性"的意思就是保证在创建单例对象的 ...
- 10.Java设计模式 工厂模式,单例模式
Java 之工厂方法和抽象工厂模式 1. 概念 工厂方法:一抽象产品类派生出多个具体产品类:一抽象工厂类派生出多个具体工厂类:每个具体工厂类只能创建一个具体产品类的实例. 即定义一个创建对象的接口(即 ...
- Java设计模式探讨之单例模式
单例模式是在平时的项目开发中比较常见的一种设计模式,使用比较普遍,网上的资料也是一抓一大把,小Alan也来凑凑热闹,为以后充实点设计模式相关的内容做个简单的开篇. 单例模式是一种创建对象的模式,用于产 ...
- Java设计模式学习01——单例模式(转)
原地址:http://blog.csdn.net/xu__cg/article/details/70182988 Java单例模式是一种常见且较为简单的设计模式.单例模式,顾名思义一个类仅能有一个实例 ...
- 【java设计模式】-04单例模式
单例模式 定义: 确保一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 类型: 创建类模式 类图: 单例模式特点 1.单例类只能有一个实例. 2.单例类必须自己创建自己的唯一实例. 3.单 ...
- Java设计模式09:单例模式的强化(控制实例个数n)
1. 单例模式的本质: 控制实例数目(目的节约资源) 2. 单例模式体现的一些思想: (1)延迟装载(Lazy Load):懒汉式 (2)缓存:饿汉式 3. 单例模式的变形使用: 控制使用实例个数为3 ...
随机推荐
- Mysql实现企业级日志管理、备份与恢复实战
背景 随着业务的发展,公司业务和规模不断扩大,网站积累了大量的用户信息和数据,对于一家互联网公司来说,用户和业务数据是根基.一旦公司的数据错乱或者丢失,对于互联网公司而言就等于说是灭顶之灾,为防止系统 ...
- netty源码分析
1.Netty是由JBOSS提供的一个java开源框架.Netty提供异步的.事件驱动的网络应用程序框架和工具,用以快速开发高性能.高可靠性的网络服务器和客户端程序.也就是说,Netty 是一个基于N ...
- mysql索引优化面试题
曾经偷偷的面试了两个单位,都提到了Mysql的优化问题,所以以后要多多学习数据库的优化知识了.建设数据库的优化大概主要就是索引的优化了吧,因为我们不可能修改数据结构的情况下,提高数据库的查询效率似乎也 ...
- javascript设计模式——策略模式
前面的话 在程序设计中,常常遇到类似的情况,要实现某一个功能有多种方案可以选择.比如一个压缩文件的程序,既可以选择zip算法,也可以选择gzip算法.这些算法灵活多样,而且可以随意互相替换.这种解决方 ...
- day6、Linux下如何找出7天以前的文件删除
有些时候,由于系统产生的日志文件,使服务器的磁盘空间紧张,所以怎么删除7天以前的日志文件及让系统只保留7天以内的日志文件 方法一 使用命令:find + |xargs + ls 命令方法:find / ...
- MySQL系列:基于binlog的增量订阅与消费(一)
在一些业务场景中,像在数据分析中我们有时候需要捕获数据变化(CDC):在数据审计中,我们也往往需要知道数据从这个点到另一个点的变化:同样在实时分析中,我们有时候需要看到某个值得实时变化等. 要解决以上 ...
- JQuery插件开发标准写法
;//step01 定义JQuery的作用域 (function ($) { //step03-a 插件的默认值属性 var defaults = { prevId: 'prevBtn', prevT ...
- javascript中的事件Event
一.事件流 1.事件流:描述的是从页面中接受事件的顺序 IE的事件流是事件冒泡流,Netscape的事件流是事件捕获流. 2.事件冒泡 IE的事件流叫做事件冒泡(event bubbing),即事件开 ...
- JavaScript 数组最大值
JavaScript 数组最大值 在js中可以使用Math.max()获取最大值. 如: console.log(Math.max("1","11"," ...
- 小白的Python之路 PEP8 代码风格
转载自http://www.douban.com/note/134971609/ Python 的代码风格由 PEP 8 描述.这个文档描述了 Python 编程风格的方方面面.在遵守这个文档的条件下 ...