Java中的GOF23(23中设计模式)--------- 单例模式(Singleton)
Java中的GOF23(23中设计模式)--------- 单例模式(Singleton)
在Java这这门语言里面,它的优点在于它本身的可移植性上面,而要做到可移植的话,本身就需要一个中介作为翻译工作,以达到本地和Java的统一,但是就这点而言就相当的消耗资源,所以就Java程序员需要不断的去优化自己的代码。今天所研究的单例模式就是在这样的条件下产生的,
所谓单例模式,就是只有一个实例,在堆里面只有一个。假如我们的实例,就需要一个,但是会多次用到,这样的话就会出现很尴尬的问题。
比如:
- Windows的TaskManager(任务管理器)就是很典型的只需要一个实例,
- Windows的Recycle(回收站)在系统中,回收站只维护一个实例
- 在我们的项目里面,会常常使用到读取配置文件的类,
- 网站的计数器,一般也采用单例模式,否则难以实现同步
- 数据库连接池设计一般也采用单例模式,因为数据库连接是一种数据库资源
- 操作系统的文件系统,因为一个操作系统只能有一个文件系统
- Application 也是单例模式
- Spring中,每个Bean默认就是单例的,这样做的优点是Spring容器可以管理
- servlet编程中,每个Servlet也是单例的
- 在Spring MVC/struts 1 中,所使用到的控制对象也是单例的
在上述里面我们了解到,单例模式在我们项目中,几乎是天天出现,所以在这里,我们仔细研究一下,这种设计模式的怎么实现最好(说到实现,它的实现我们大多数人只知道有两种,而还有三种模式知道的人不是很多,以及利用反序列化,反射漏洞去强制解除单例)
- 饿汉模式
- 使用static属性来保持对象的单例模式,但是必须在类加载的时候加载,所以没有延迟实例化的性能
* 而在得到实例的对象中,没有对资源的同步锁,所以调用效率高
* 而使用JVM类加载器,JVM底层天然是线程安全的, - 优点:线程安全调用率较高
- 缺点:不能延迟加载
public class Eager_Singleton { private static Eager_Singleton singleton = new Eager_Singleton(); private Eager_Singleton(){} public static Eager_Singleton newInstance(){
return singleton;
}
}
- 使用static属性来保持对象的单例模式,但是必须在类加载的时候加载,所以没有延迟实例化的性能
- 懒汉模式
- 依旧使用static属性来保持对象的单例模式,但是在方法里面new出来,当我们去调方法的时候,再去加载,就是懒人,懒得一上来就加载,但是就因为在方法里面实例化,所以我们要在该方法上面使用锁的概念来对他进行同步处理,如果不加锁,那我们在A线程里面去掉用它和在B线程里面去调用它他就不是一个概念了
- 优点:可以延时加载,并且线程是安全的
- 缺点:就是方法实现了同步,所以资源的利用率比较低。
- 依旧使用static属性来保持对象的单例模式,但是在方法里面new出来,当我们去调方法的时候,再去加载,就是懒人,懒得一上来就加载,但是就因为在方法里面实例化,所以我们要在该方法上面使用锁的概念来对他进行同步处理,如果不加锁,那我们在A线程里面去掉用它和在B线程里面去调用它他就不是一个概念了
/**
* 懒汉单例模式
* @author 刘酸酸
*
*/
public class Sluggard_Singleton { private static Sluggard_Singleton singleton = null; private Sluggard_Singleton(){} public synchronized static Sluggard_Singleton newInstance(){
if(singleton == null){
singleton = new Sluggard_Singleton();
}
return singleton;
}
}
- 双重检测式
- 将同步块内部下方的代码放至到IF内部
- 所面临的问题:由于编译器优化等原因JVM底层内部模型的原因,偶尔会出问题,不建议使用。
/**
* 双重检查锁实现单例模式
* @author 刘酸酸
*
*/
public class DoubleCheck_Singleton { private static DoubleCheck_Singleton instance = null; public static DoubleCheck_Singleton getInstance() {
if (instance == null) {
DoubleCheck_Singleton sc;
synchronized (DoubleCheck_Singleton .class) {
sc = instance;
if (sc == null) {
synchronized (DoubleCheck_Singleton .class) {
if(sc == null) {
sc = new DoubleCheck_Singleton ();
}
}
instance = sc;
}
}
}
return instance;
} private DoubleCheck_Singleton () { } }
静态内部类方式实现
- 该方式是结合懒汉式和饿汉式的优点,
/**
* 测试静态内部类实现单例模式
* 这种方式:线程安全,调用效率高,并且实现了延时加载!
* @author 刘酸酸
*
*/
public class StaticInnerClass{ private static class StaticInnerClass{
private static final StaticInnerClass instance = new StaticInnerClass();
} private StaticInnerClass(){
} //方法没有同步,调用效率高!
public static StaticInnerClassgetInstance(){
return StaticInnerClass.instance;
} }
- 枚举实现
- 在Java的JVM里面就是一种天然的单例模式,那就是枚举类型,如果使用枚举类型的话会受到JVM底层的保护
比如使用反序列化,和使用反射是不能打破这个原则的
优点:上诉
缺点:不能延迟加载
- 在Java的JVM里面就是一种天然的单例模式,那就是枚举类型,如果使用枚举类型的话会受到JVM底层的保护
/**
* 测试枚举式实现单例模式(没有延时加载)
* @author 刘酸酸
*
*/
public enum Enum_Singleton{ //这个枚举元素,本身就是单例对象!
INSTANCE; //添加自己需要的操作!
public void xxxxx(){
}
}
- 如何防止反序列化,反射破环单例模式(枚举除外)
- 在反射中如果我们调用了私有的构造器的话,我们就会抛异常,但是会有人跳过合法检测,这样是可以访问到我们的私有构造器的
import java.lang.reflect.Constructor; /**
* 测试反射和反序列化破解单例模式
* @author 刘酸酸
*
*/
public class Main{ public static void main(String[] args) throws Exception { //通过反射的方式直接调用私有构造器
Class<Singleton> clazz = (Class<Singleton>) Class.forName("com.suansuan.singleton.Singleton");
Constructor<Singleton> c = clazz.getDeclaredConstructor(null);
//跳过合法检查
c.setAccessible(true);
Singleton s1 = c.newInstance();
Singleton s2 = c.newInstance();
System.out.println(s1);
System.out.println(s2);
}
}
- 反序列化时,我们也不能得到一个对象,所以,下述代码解决这两种问题
/**
* 测试懒汉式单例模式(如何防止反射和反序列化漏洞)
* @author 刘酸酸
*
*/
public class Singleton implements Serializable {
//类初始化时,不初始化这个对象(延时加载,真正用的时候再创建)。
private static Singleton instance; private Singleton (){ //私有化构造器
//反射时,我们作如下操作既可以规避,跳过合法检测的非法访问
if(instance!=null){
throw new RuntimeException();
}
} //方法同步,调用效率低!
public static synchronized Singleton getInstance(){
if(instance==null){
instance = new Singleton ();
}
return instance;
} //反序列化时,如果定义了readResolve()则直接返回此方法指定的对象。而不需要单独再创建新对象!
private Object readResolve() throws ObjectStreamException {
return instance;
} }
- 反序列化时,我们也不能得到一个对象,所以,下述代码解决这两种问题
总结:使用枚举去替代饿汉式,使用静态内部类去替代懒汉式
第一次写博客,谢谢大家如果有不对的房指出来,我们共同进步,共同发展,谢谢大家
Java中的GOF23(23中设计模式)--------- 单例模式(Singleton)的更多相关文章
- Spring中常用的23中设计模式
1.spring 中常用的设计模式有23中 分类 设计模式 创建型 工厂方法模式(FactoryMethod).抽象工厂模式(AbstractFactory).建造者模式(Builder).原型 ...
- Java中的GOF23(23中设计模式)--------- 工厂模式(Factory)
Java中的GOF23(23中设计模式)--------- 工厂模式(Factory) 在给大家介绍工厂模式之前,我想和大家聊聊面向对象的那点事,在这里,引入三个概念. 开闭原则(Open Close ...
- java中的23中设计模式(转)
设计模式(Design Patterns) --可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了 ...
- java开发中的23中设计模式详解--大话设计模式
设计模式(Design Patterns) ——可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了 ...
- java中的23中设计模式(转载的,有时间一定要熟读)
设计模式(Design Patterns) --可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了 ...
- java开发中的23中设计模式
设计模式(Design Patterns) ——可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了 ...
- Java开发中的23中设计模式详解(一)工厂方法模式和抽象工厂模式
一.设计模式的分类 总体来说设计模式分为三大类: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式. 结构型模式,共七种:适配器模式.装饰器模式.代理模式.外观模式.桥接 ...
- java开发的23中设计模式
本文转自 Java开发中的23种设计模式详解(转) 设计模式(Design Patterns) ——可复用面向对象软件的基础 设计模式(Design pattern)是一套被反复使用.多数人知晓 ...
- java的23中设计模式
一.设计模式的分类 总体来说设计模式分为三大类: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单例模式.建造者模式.原型模式. 结构型模式,共七种:适配器模式.装饰器模式.代理模式.外观模式.桥接 ...
随机推荐
- 美团(iPad)顶部界面的简单实现, 及开发时常见bug
项目功能介绍:1.支持横竖屏旋转,界面正常显示2.通过点击界面顶部"美团",可展示出左右双tableView分别显示服务类列表和子类列表3.通过点击界面顶部"广州&quo ...
- 【WP 8.1开发】解决摄像头翻转问题(RuntimeApp篇)
昨天,我非常马虎地给大家说了有关处理物理摄像头翻转的话题,今天,还是这个话题,而且内容不差,只是为了完整性,顺便也提供了运行时API的版本,其实实现起来与SL框架版本差不多,毕竟这两个框架都有不少AP ...
- 你的 mixin 兼容 ECMAScript 5 吗
原文:Are your mixins ECMAScript 5 compatible? 作者:Nicholas C. Zakas 我最近在与客户合作的项目中,需要充分利用的 ECMAScript 5, ...
- C#多线程之旅(3)——线程池
v博客前言 先交代下背景,写<C#多线程之旅>这个系列文章主要是因为以下几个原因:1.多线程在C/S和B/S架构中用得是非常多的;2.而且多线程的使用是非常复杂的,如果没有用好,容易造成很 ...
- 总结整理 -- ruby系列
基础学习 ruby -- 基础学习(一)项目文件夹说明 ruby -- 基础学习(二) 外键配置实现级联删除 ruby -- 基础学习(三)设置中国时区时间 ruby -- 基础学习(四)TimeDa ...
- 2014 -> 2015
2014年初在公司的发展不太顺利, 发现比好多小伙伴的发展速度都要慢了,钱不多,职位也不高,做的事情成长也不快. 为了职业发展考虑,年中就一直想换一个好一点的工作机会, 年中拿了好几个offer, 有 ...
- Windows 8(虚拟机环境)安装.NET Framework3.5(includes .NET 2.0 and 3.0)
按照这篇文章:http://blogs.technet.com/b/aviraj/archive/2012/08/04/windows-8-enable-net-framework-3-5-inclu ...
- Spring MVC异常处理详解
Spring MVC中异常处理的类体系结构 下图中,我画出了Spring MVC中,跟异常处理相关的主要类和接口. 在Spring MVC中,所有用于处理在请求映射和请求处理过程中抛出的异常的类,都要 ...
- Windows Azure Virtual Machine (26) 使用高级存储(SSD)和DS系列VM
<Windows Azure Platform 系列文章目录> Update: 2016-11-3,如果大家在使用Linux VM,使用FIO进行IOPS测试的时候,请使用以下命令: su ...
- C# 控制台或者winform程序开启http的监听状态
1 public class THttpListener { HttpListener listerner; /// <summary> /// /// </summary> ...