1.饿汉式

  1. public class SingletonDemo {
  2. private static SingletonDemo s=new SingletonDemo();
  3. private SingletonDemo() {}
  4. public static SingletonDemo getInstance() {
  5. return s;
  6. }
  7. }
  1. public class Test {
  2. public static void main(String[] args) {
  3. SingletonDemo s=SingletonDemo.getInstance();
  4. }
  5. }

步骤:    1.new出静态对象

    2.构造方法私有化

    3.写一个方法返回静态对象

说明:   1.由于加载类时天然的是线程安全

    2.方法没有同步,调用效率高

    3.立即加载,没有延时加载的优势

2.懒汉式(延时加载)

  1. public class SingletonDemo {
  2. private static SingletonDemo s;
  3. private SingletonDemo() {}
  4. public static synchronized SingletonDemo getInstance() {
  5. if(s==null) s=new SingletonDemo();
  6. return s;
  7. }
  8. }
  1. public class Test {
  2. public static void main(String[] args) {
  3. SingletonDemo s=SingletonDemo.getInstance();
  4. }
  5. }

优势:   1.修改点同步,在线程高并发时,能够保证安全性

    2.延时new出类,能够做到用的时候去new它

3.双重检测锁实现

由于java编译器优化的原因和JVM底层内部模型的原因,偶尔会出一点问题,不建议使用

这个模式将同步内容下方到if内部,提高了执行效率,不必每次获取对象时都进行同步,至于一次才同步

  1. public class SingletonDemo {
  2. private static SingletonDemo instance;
  3. private SingletonDemo() {}
  4. public static SingletonDemo getInstance() {
  5. if(instance==null) {
  6. SiSingletonDemo sc;
  7. synchronized (SingletonDemo.class) {
  8. sc=instance;
  9. }
  10. if(sc==null) {
  11. synchronized (SingletonDemo.class) {
  12. if(sc==null) {
  13. sc=new SingletonDemo();
  14. }
  15. }
  16. instance=sc;
  17. }
  18. }
  19. return instance;
  20. }
  21. }

4.静态内部类(懒加载)

  1. public class SingletonDemo {
  2. private static class SingletonClassInstance{
  3. private static final SingletonDemo instance=new SingletonDemo();
  4. }
  5. private SingletonDemo() {}
  6. public static SingletonDemo getInstance() {
  7. return SingletonDemo.getInstance();
  8. }
  9. }

5.枚举单例模式

  1. public enum SingletonDemo {
  2. INSTANCE; //枚举本来就是单例对象
  3. //避免了反序列和反射的调用
  4. //缺点,没有延时调用
  5. public void singletonOperation() {
  6.  
  7. }
  8. }

6.统一建模语言UML(unified modeling language)

可以拖动类做uml图

利用反射破解单例设计模式(不包含枚举单例,因为枚举型单例是利用JVM底层实现的单例设计模式)

  1. public class TestSingleObject {
  2. public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
  3. SingleObject s1=SingleObject.getInstance();
  4. SingleObject s2=SingleObject.getInstance();
  5.  
  6. Class<SingleObject> clazz=(Class<SingleObject>) Class.forName("com.littlepage.singletonPattern.SingleObject");
  7. Constructor<SingleObject> c=clazz.getDeclaredConstructor(null);
  8. c.setAccessible(true);//跳过检测
  9. SingleObject s3=c.newInstance();
  10. SingleObject s4=c.newInstance();
  11. System.out.println(s1);
  12. System.out.println(s2);
  13. System.out.println(s3);
  14. System.out.println(s4);
  15. }
  16. }

解决方法,在空参构造中加入

if(SingletonObject!=null){

throw new runtimeExpection();

}

进行调用时抛出异常就行

  1. //举例
  2. public class SingleObject {
  3. public static class SingleObjectInstance{
  4. public static final SingleObject instance=new SingleObject();
  5. }
  6. private SingleObject(){
  7. if(SingleObjectInstance.instance!=null){
  8. try {
  9. throw new Exception("运行时异常");
  10.  
  11. } catch (Exception e) {
  12. e.printStackTrace();
  13. System.exit();
  14. }
  15. }
  16. }
  17. public static SingleObject getInstance(){
  18. return SingleObjectInstance.instance;
  19. }
  20. }

利用序列化反序列化进行破解

  1. public class TestSingleObject {
  2. public static void main(String[] args) throws IOException, ClassNotFoundException {
  3. SingleObject s1=SingleObject.getInstance();
  4. System.out.println(s1);
  5. FileOutputStream fops=new FileOutputStream("d:/a.txt");
  6. ObjectOutputStream oops=new ObjectOutputStream(fops);
  7. oops.writeObject(s1);
  8. oops.close();
  9.  
  10. ObjectInputStream ois =new ObjectInputStream(new FileInputStream("d:/a.txt"));
  11. SingleObject s2=(SingleObject) ois.readObject();
  12. System.out.println(s2);
  13. ois.close();
  14.  
  15. }
  16. }

解决办法:

  1. private Object resdResolve() throws ObjectStreamException{
  2. return SingleObjectInstance.instance;
  3. }

添加该代码,可以使读取Object时直接返回已经存在的实例

设计模式(三)Singleton Pattern单例设计模式的更多相关文章

  1. Java面试 - 什么是单例设计模式,为什么要使用单例设计模式,如何实现单例设计模式(饿汉式和懒汉式)?

    什么是单例设计模式? 单例设计模式就是一种控制实例化对象个数的设计模式. 为什么要使用单例设计模式? 使用单例设计模式可以节省内存空间,提高性能.因为很多情况下,有些类是不需要重复产生对象的. 如果重 ...

  2. 【工作中学习1】两个设计模式:Singleton(单例)和 Adapter(适配器)

    好久没有写自己的学习小博客,罪过罪过..最近本菜鸟在项目中接触到经常用到的设计模式,首先是Singleton(单例),这个相信大家都会用到很多,所以自己用代码实现一下,有助于自己学习理解,如有不对,请 ...

  3. GOF设计模式之1:单例设计模式

    1.单例设计模式核心作用: 保证一个类只有一个实例,并且提供了访问该实例的全局访问点 2.常见应用场景: window的任务管理器 项目中读取配置文件一般也是一个单例模式 数据库连接池的设计也是采用单 ...

  4. IOS设计模式第二篇之单例设计模式

    现在我们的组件已经有组织了.你需要从其他的地方得到数据,你也可以创建一个API类管理数据这个下个设计模式单例里面介绍. 这个单例设计模式确保这个类仅仅拥有一个实例,并且为这个实例提供一个全局的访问点. ...

  5. 初学者学Java设计模式(一)------单例设计模式

    单例设计模式 单例设计模式是指一个类只会生成一个对象,优点是他可以确保所有对象都访问唯一实例. 具体实现代码如下: public class A { public static void main(S ...

  6. 【设计模式】Java之单例设计模式

    1.单例设计模式:一个类只能有一个对象 1.1 创建单例类的步骤: 1.将构造方法私有化 2.创建私有的静态成员变量 3.共有的静态成员方法,提供当前的唯一对象 1.1 创建单例的两种方式: 1.饿汉 ...

  7. Java Notes 00 - Singleton Pattern(单例总结)

    转:http://hukai.me/java-notes-singleton-pattern/ 这里不赘述单例模式的概念了,直接演示几种不同的实现方式. 0)Eager initialization ...

  8. 设计模式课程 设计模式精讲 8-10 单例设计模式-ThreadLocal线程单例

    1 课程讲解 1.1 应用场景 2 代码演练 2.1 threadLocal应用 1 课程讲解 1.1 应用场景 多线程的时候: 使用同步锁使用时间换空间的方式,(线程排队时间比较长) 而使用thre ...

  9. 设计模式课程 设计模式精讲 8-8 单例设计模式-Enum枚举单例、原理源码解析以及反编译实战

    1 课堂解析 2 代码演练 2.1 枚举类单例解决序列化破坏demo 2.2 枚举类单例解决序列化破坏原理 2.3 枚举类单例解决反射攻击demo 2.4 枚举类单例解决反射攻击原理 3 jad的使用 ...

随机推荐

  1. The Little Prince-12/11

    The Little Prince-12/11 最后一段话!!!hha,傻傻的我们...... 成人们对数字情有独钟.如果你为他们介绍一个朋友,他们从不会问你“他的嗓子怎么样?他爱玩什么游戏?他会采集 ...

  2. getopt_long_only

    st_i2c_parser.c中用到,遂学习一下: 参考:https://blog.csdn.net/pengrui18/article/details/8078813 https://blog.cs ...

  3. centos系统swap设置 查看swap分区的方法

    交换分区swap,意思是“交换”.“实物交易”,它的功能就是在内存不够的情况下,操作系统先把内存中暂时不用的数据,存到硬盘的交换空间,腾出内存来让别的程序运行,和Windows的虚拟内存(pagefi ...

  4. python基础:re模块匹配时贪婪和非贪婪模式

    python贪婪和非贪婪 正则表达式通常用于在文本中查找匹配的字符串.Python里数量词默认是贪婪的(在少数语言里也可能是默认非贪婪),总是尝试匹配尽可能多的字符:非贪婪则相反,总是尝试匹配尽可能少 ...

  5. pxc集群进入非主模式怎么让最后的节点允许提供服务

    这种情况一般是,集群其他节点意外宕机而最后一个节点无法自我仲裁,而进入非主模式. 该模式拒绝任何SQL的执行: ERROR 1047 (08S01): WSREP has not yet prepar ...

  6. Django Form&ModelForm

    ModelForm: 首先导入所需模块 from django.forms import ModelFormfrom django.forms import widgets as form_widge ...

  7. HTTP小幺鸡接口管理工具安装与配置说明

    http://www.xiaoyaoji.cn/doc/TxybXPTdx 小幺鸡接口管理工具安装说明 使用可以参考:https://blog.csdn.net/qincidong/article/d ...

  8. API网关性能比较:NGINX vs. ZUUL vs. Spring Cloud Gateway vs. Linkerd(转)

    前几天拜读了 OpsGenie 公司(一家致力于 Dev & Ops 的公司)的资深工程师 Turgay Çelik 博士写的一篇文章(链接在文末),文中介绍了他们最初也是采用 Nginx 作 ...

  9. Net SMTP QQ 发送邮件

    调用DEMO var currUser = new List<string> { "123@qq.com" , "123@qq.com" , &qu ...

  10. 【题解】Luogu P2146 [NOI2015]软件包管理器

    题面:https://www.luogu.org/problemnew/lists?name=2146 这道题要用树链剖分,我博客里有对树链剖分的详细介绍 这道题就是树链剖分的模板,详细解释见程序. ...