酷站(www.ku0.com)-致力于为互联网从业者提供动力!

热门关键词:  企业  as  baidu  c4rp3nt3r  美女
阿里云爆款特惠,精选爆款产品低至0.55折
酷站

Java中Singleton的3种实现方式

来源:转载 作者:秩名 人气: 发布时间:2021-02-21
本篇文章主要介绍了Java中Singleton的3种实现方式,对大家的学习或者工作具有一定的参考学习价值,感兴趣的小伙伴们可以参考一下,也感谢大家对酷站(ku0.com)的支持。

一、什么是Singleton?

设计模式》的作者、Eclipse和 Junit 的开发者 Erich Gamma 在它的理论体系中将 Singleton 定义为仅仅被实例化一次的类。在当今面向对象程序的实际开发中,Singleton 通常被用来代表一个无状态的对象,例如函数和那些本质上唯一的系统组件。

值得注意的是,使类成为 Singleton 会使得它的客户端测试变得非常困难,因为我们不可能给Singleton替换模拟实现,除非我们实现一个充当其类型的接口。

实现 Singleton 有三种常见方法,他们或是保持构造器私有并导出公有的静态成员,或是声明一个包含单个元素的枚举类型。

二、Singleton实现 —— 构造器私有

1、公有静态成员为一个final域

//Singleton with public final field
public class Elvis {
 public static final Elvis INSTANCE = new Elvis(); 
 pritvate Elvis() { ... }
 public void leaveTheBuilding() { ... }
}
 

在这个类中,我们仅仅拥有一个私有的构造器,它也只在初始化final域时被调用一次。由于缺少可以使用的构造器,后续的程序无法再创建 Elvis 对象。这保证了在该Java程序的整个生命周期中, Elvis 对象有且只有一个存在。

但需要注意的是,一些高权限的客户端可以借助 AccessibleObject.setAccessible 方法通过反射机制调用私有的构造器。为了避免这样的可能的攻击,可以修改构造器,让它在被要求创建第二个实例的时候抛出异常。

公有域方法的主要优势在于,API很清楚地表明了这个类是一个 Singleton ,毕竟这是一个公有的静态属性。另外,这个方法要更加简单。

2、公有静态成员为一个静态工厂方法

//Singleton with static factory
public class Elvis {
 private static final Elvis INSTANCE = new Elvis();
 pritvate Elvis() { ... }
 public static Elvis getInstance(){ return INSTANCE; }
 public void leaveTheBuilding(){ ... }
}
 

显然,无论怎样调用 getInstance 方法,返回的都是同一个对象的引用。注意上面提示的反射攻击问题依然存在。

静态工厂方法有三大优势

  • 第一,它提供了更多的灵活性,在不改变API的前提下,我们可以轻易地自由调整这个类是否是Singleton。工厂方法返回该类的唯一实例,但它很容易修改成别的样子,例如为每个调用该方法的线程提供唯一实例。
  • 第二,如果程序需要,我们可以编写一个泛型 Singleton 工厂。
  • 第三,我们可以通过方法引用作为提供者,比如 Elvis::instance 就是一个 Supplier< Elvis >

(注:方法引用是Java8的一个新特性)

除非我们需要上述的其中一种优势,我们还是应该选择更简单易懂的使用公有域的方法。

3、将利用上述方法实现的Singleton类变为可序列化的

使用上述两种方法实现的 Singleton ,要把他们变成可序列化的,不能仅仅在声明中加上 implements Serializable 。为了维护并保证 Singleton ,我们必须生命所有实例域都是瞬时的,并提供一个 readResolve 方法。否则在我们每次序列化时都会创建一个新的实例。为了防止这种情况,我们要在 Elvis 类中加入如下这样的 readResolve 方法。

//readResolve method to preserve singleton property
 private Object readResolve(){
 //Return the one true Elvis and let the garbage collector take care of the Elvis impersonator
 return INSTANCE;
 }
 

三、Singleton实现 —— 声明包含单个元素的枚举类型
 
//Enum singleton - the preferred approach
public enum Elvis{
 INSTANCE;
 public void leaveTheBuilding(){ ... }
}
 

这种方法在功能上与公有域方法相似,但更加简洁,无偿地提供了序列化机制,绝对防止多次实例化,即使是在面对复杂的序列化或者反射攻击的时候。 虽然这种方法还没有广泛采用,但是单元素的枚举类型经常成为实现 Singleton 的最佳方法。 注意,如果 Singleton 必须扩展一个超类,而不是扩展 Enum 的时候,则不宜使用这个方法(虽然可以声明枚举去实现接口)。

版权声明:本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 959677720#qq.cn(#换@) 举报,一经查实,本站将立刻删除。
原文链接:https://www.cnblogs.com/WangXianSCU/p/14423665.html

相关文章

  • 解决idea中@Data标签getset不起作用的问题

    解决idea中@Data标签getset不起作用的问题

    spring cloud中使用@Data标签,不用手动添加get set方法,但是如果项目中其他类中使用getset方法,如果报错,原因是idea中没有添加Lombok插件,添加上插件便可以解决。 截图如下 补充:Intellij IDEA无法使用lombok、@Data不起作用,set......
    02-23
  • Java中Singleton的3种实现方式

    Java中Singleton的3种实现方式

    一、什么是Singleton? 《设计模式》的作者、Eclipse和 Junit 的开发者 Erich Gamma 在它的理论体系中将 Singleton 定义为仅仅被实例化一次的类。在当今面向对象程序的实际开发中,Singleton 通常被用来代表一个无状态的对象,例如函数和......
    02-21
  • Java动态加载类详解

    Java动态加载类详解

    在讲解动态加载类之前呢,我们先弄清楚为什么要动态加载类,静态加载不行吗?我们可以看下面的实例: 我在文件夹里写了Office.java 类和 Word.java类,如下: Office.java class Office{ public static void main(String[] args){ if(arg......
    02-21
  • 解决redisTemplate中leftPushAll隐性bug的问题

    解决redisTemplate中leftPushAll隐性bug的问题

    前言 请看下面代码: String key = String.format(test_key:%s, System.currentTimeMillis()/1000); String key2=key+_2; String key3=key+_3; ListString t1=new ArrayList(); t1.add(2); t1.add(3); t1.add(4); t1.add(5); t1.add(1); ......
    02-17
  • 解决Intellij IDEA覆盖tomcat配置的问题

    解决Intellij IDEA覆盖tomcat配置的问题

    刚刚遇到bug,每次修改自己环境里tomcat的server.xml,再次在Intellij里启动tomcat之后,修改就消失了,心态很炸裂hhh 研究了好久,最终把目光放在了下图所示的地方。 一开始以为是自己环境变量没配,但是看了一下发现没问题,又在cmd里......
    02-12
  • Java使用FTP上传文件并模拟接受的教程

    Java使用FTP上传文件并模拟接受的教程

    0. 导入相关依赖 !-- FTP --dependency groupIdcommons-net/groupId artifactIdcommons-net/artifactId version3.6/version /dependency !-- Hutool:一个小而全的Java工具类库,具体介绍参照附录1 -- dependency groupIdcn.hutool/groupI......
    02-12
  • spring security认证异常后返回中文提示的问题详解

    spring security认证异常后返回中文提示的问题详解

    1.加载中文提示类 @Configurationpublic class ReloadMessageConfig { /** * 加载中文的认证提示信息 * * @return */ @Bean public ReloadableResourceBundleMessageSource messageSource() { ReloadableResourceBundleMessageSource mes......
    02-04
  • java中i = i++和i =++i的深入详解

    java中i = i++和i =++i的深入详解

    public class Count { public static void main(String[] args) { int i = 0; i = i++ ; System.out.println(i); }} 上面代码输出的i为0,如果是把i = i++换成i=++i,又会输出1,这是由于i++是先赋值,再计算导致,但是为什么先赋值呢? ......
    02-01
  • springboot使用CommandLineRunner解决项目启动时初始化资源

    springboot使用CommandLineRunner解决项目启动时初始化资源

    前言: 在我们实际工作中,总会遇到这样需求,在项目启动的时候需要做一些初始化的操作,比如初始化线程池,提前加载好加密证书等。 今天就给大家介绍一个 Spring Boot 神器,专门帮助大家解决项目启动初始化资源操作。 这个神器就是Comm......
    02-01
  • Java为何需要平衡方法调用与内联

    Java为何需要平衡方法调用与内联

    在 Java 中,方法调用一般通过 Virtual Call 还有 Classic Call。 Classic Call 就是直接指向方法的地址,需要一次寻址到方法的地址,比直接执行代码慢。 Virtual Call 需要通过 VMT(Virtual Method Table)。这个VMT存储的是该class对象......
    01-29

最新更新