JAVA中单例模式,java设计模式之单例模式

  JAVA中单例模式,java设计模式之单例模式

  00-1010一、什么是单体模式二。singleton模式用在哪里?单例模式IV的优点和缺点。手写单例模式饿-中文枚举饿-中文DCL懒-中文双检锁懒-中文内部懒汇总

  00-1010单例模式是Java中最简单的设计模式之一。这种类型的设计模式属于创造性模式,它提供了创建对象的最佳方式。

  这种模式涉及一个单独的类,它负责创建自己的对象,同时确保只创建一个对象。该类提供了访问其唯一对象的方法,可以直接访问该对象,而无需实例化该类的对象。

  注意

  单一模式只能有一个实例对象。单例类必须创建自己唯一的实例。singleton类必须向所有其他对象提供这个实例。

  00-1010单例模式通常用于需要实例的程序中,例如

  1.1的IOC容器。Spring框架用于singleton模式,默认对象创建时间为singleton模式。

  2.2的后端。ResultBean统一返回前端封装类,在项目中是唯一的。只有一个对象用于将JSON返回到前端进行渲染。

  例如,在JDK也有独生子女模式

  运行时体现了饥饿-中文单例控制台体现了EmptyNavigableSet内部的懒惰式单例,在懒惰式单例集合中带有双重检查锁。Reverse _ order内部类似懒惰的单例比较器。naturalrordercomprator . instance枚举饥饿的中国单例。

  

目录

 

  00-1010提供对唯一实例的访问。

  它可以节省系统资源,提高系统性能,减少不必要的内存开销。

  允许可变数量的实例(多实例类)

  00-1010扩展困难(缺少抽象层)

  单班的责任太重了。

  由于自动垃圾收集机制,共享单例对象的状态可能会丢失。

  

一、什么是单例模式

 

  

二、哪些地方用到了单例模式

包com . wanshi . single;//Hungry-中文单实例公有类Hungry {//会造成资源浪费,占用CPU私有字节[]data1=new byte[1024 * 1024];私有字节[]数据2=新字节[1024 * 1024];私有字节[]数据3=新字节[1024 * 1024];私有字节[]数据4=新字节[1024 * 1024];private Hungry(){ system . out . println( Hungry init . );}私静饿饿=新饿();public static Hungry getInstance(){ return Hungry;} } class Test { public static void main(String[]args){ Hungry Hungry=Hungry . getinstance();hungry hungry 2=hungry . getinstance();System.out.println(饿了么);system . out . println(hungry 2);}}

 

  

三、单例模式的优缺点

包com . wanshi . single;导入Java . lang . reflect . constructor;//enum是class类公共enum EnumSingle { INSTANCEpublic static enum single getInstance(){ return INSTANCE;} }类测试{ public static void main(String[]args)抛出异常{ enum single instance 1=enum single。实例;ConstructorEnumSingle declared constructor=enum single . class . getdeclaredconstructor(string . class,int。

 

  class); declaredConstructor.setAccessible(true); EnumSingle instance2 = declaredConstructor.newInstance(); System.out.println(instance1); System.out.println(instance2); }}在这里自行下载jad编译工具即可

  

 

  枚举类最后反编译源码

  jad工具反编译

  jad -sjava EnumSingle.class

  

// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.// Jad home page: http://www.kpdus.com/jad.html// Decompiler options: packimports(3) // Source File Name: EnumSingle.javapackage com.wanshi.single;public final class EnumSingle extends Enum{ public static EnumSingle[] values() { return (EnumSingle[])$VALUES.clone(); } public static EnumSingle valueOf(String name) { return (EnumSingle)Enum.valueOf(com/wanshi/single/EnumSingle, name); } private EnumSingle(String s, int i) { super(s, i); } public static EnumSingle getInstance() { return INSTANCE; } public static final EnumSingle INSTANCE; private static final EnumSingle $VALUES[]; static { INSTANCE = new EnumSingle("INSTANCE", 0); $VALUES = (new EnumSingle[] { INSTANCE }); }}

 

  

DCL懒汉式

package com.wanshi.single;public class Lazy { private static Lazy lazy; public static Lazy getInterface() { synchronized (Lazy.class) { if (lazy == null) { lazy = new Lazy(); } } return lazy; } public static void main(String[] args) { Lazy lazy = Lazy.getInterface(); Lazy lazy2 = Lazy.getInterface(); System.out.println(lazy); System.out.println(lazy2); }}

 

  

双检锁懒汉式

package com.wanshi.single;import java.lang.reflect.Constructor;import java.lang.reflect.Field;public class LazyMan { private static boolean flag = false; private LazyMan() { synchronized (this) { if (!flag) { flag = true; } else { throw new RuntimeException("不要试图通过反射破坏对象"); } } } private volatile static LazyMan lazyMan; //双重检查锁,懒汉式(DCL懒汉式) public static LazyMan getInstance() { if (lazyMan == null) { synchronized (LazyMan.class) { if (lazyMan == null) { //不是原子性操作,1.分配内存空间,2.执行构造方法,3.把对象指向这个空间 指令重排可能会发生 加上volatile关闭指令重排 lazyMan = new LazyMan(); } } } return lazyMan; } public static void main(String[] args) throws Exception {// LazyMan lazyMan1 = LazyMan.getInstance(); Field flag = LazyMan.class.getDeclaredField("flag"); flag.setAccessible(true); Constructor<LazyMan> declaredConstructor = LazyMan.class.getDeclaredConstructor(null); declaredConstructor.setAccessible(true); LazyMan lazyMan1 = declaredConstructor.newInstance(); flag.set(lazyMan1, false); LazyMan lazyMan2 = declaredConstructor.newInstance(); System.out.println(lazyMan1); System.out.println(lazyMan2); }}

为什么要使用 volatile 关键字呢

 

  不是原子性操作

  1.分配内存空间

  2.执行构造方法

  3.把对象指向这个空间

  指令重排可能会发生 加上volatile关闭指令重排

  

 

  

内部类懒汉式

package com.wanshi.single;public class Holder { private Holder() { } public static class InnerClass { private static final Holder HOLDER = new Holder(); }}

案例全部通过测试!

 

  

 

  

小结

单例模式共有5种创建方式,分别为饿汉式、DCL懒汉式、双检锁懒汉式、Enum枚举饿汉式,内部类懒汉式,这几种方式要掌握,项目中对于全局唯一的对象将其封装为单例模式,开箱即用,非常方便,以及面试中,会让手写单例模式,可谓是大厂必备!

 

  以上就是详解Java单例模式的实现与原理剖析的详细内容,更多关于Java单例模式的资料请关注盛行IT其它相关文章!

郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。

留言与评论(共有 条评论)
   
验证码: