经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » 设计模式 » 查看文章
Java单例模式实现,一次性学完整,面试加分项
来源:cnblogs  作者:源点+  时间:2021/4/12 9:54:53  对本文有异议

单例模式是设计模式中使用最为普遍的一种模式。属于对象创建模式,它可以确保系统中一个类只产生一个实例。这样的行为能带来两大好处:

  • 对于频繁使用的对象,可以省略创建对象所花费的时间,这对于那些重量级对象而言,是非常可观的一笔系统开销。
  • 由于new操作的次数减少,因而对系统内存的使用频率也会降低,这将减轻GC压力,缩短GC停顿时间。

在实际应用中,很多时候有一些对象我们只需要一个,例如:线程池(threadpool)、缓存(cache)、注册表(registry)、日志对象等等,这个时候把它设计为单例模式是最好的选择。

1、单例模式6种实现方法

1)懒汉模式(线程不安全)

  1. public class Singleton01 {
  2. private static Singleton01 instance;
  3. /**
  4. * 私有构造方法
  5. */
  6. private Singleton01(){}
  7. public static Singleton01 getInstance() {
  8. if(instance == null) {
  9. instance = new Singleton01();
  10. }
  11. return instance;
  12. }
  13. }

这种写法实现延迟加载,但线程不安全。禁止使用!

2)懒汉模式(线程安全)

  1. public class Singleton02 {
  2. private static Singleton02 instance;
  3. /**
  4. * 私有构造方法
  5. */
  6. private Singleton02(){}
  7. public static synchronized Singleton02 getInstance() {
  8. if(instance == null) {
  9. instance = new Singleton02();
  10. }
  11. return instance;
  12. }
  13. }

这种写法实现延迟加载,且增加synchronized来保证线程安全,但效率太低。不建议使用

3)懒汉模式(双重校验锁)

  1. public class Singleton03 {
  2. private volatile static Singleton03 instance;
  3. /**
  4. * 私有构造方法
  5. */
  6. private Singleton03(){}
  7. public static Singleton03 getInstance() {
  8. if(instance == null) {
  9. synchronized (Singleton03.class) {
  10. if (instance == null) {
  11. instance = new Singleton03();
  12. }
  13. }
  14. }
  15. return instance;
  16. }
  17. }

使用到了volatile机制。这个是第二种方式的升级版,俗称双重检查锁定。既保证了效率,又保证了安全。

4)饿汉模式

  1. public class Singleton03 {
  2. private static Singleton03 instance = new Singleton03();
  3. /**
  4. * 私有构造方法
  5. */
  6. private Singleton03(){}
  7. public static synchronized Singleton03 getInstance() {
  8. return instance;
  9. }
  10. }

这种基于类加载机制避免了多线程的同步问题,初始化的时候就给装载了。但却没了懒加载的效果。
这也是最简单的一种实现。

5)静态内部类

  1. public class Singleton04 {
  2. // 静态内部类
  3. private static class SingletonHolder {
  4. private static final Singleton04 INSTANCE = new Singleton04();
  5. }
  6. /**
  7. * 私有构造方法
  8. */
  9. private Singleton04(){}
  10. public static Singleton04 getInstance() {
  11. return SingletonHolder.INSTANCE;
  12. }
  13. }

这种方式当Singleton04类被加载时,其内部类并不会被加载,所以单例类INSTANCE不会被初始化。
只有显式调用getInstance方法时,才会加载SingletonHolder,从而实例化INSTANCE
由于实例的建立是在类加载时完成,所以天生线程安全。因此兼备了懒加载和线程安全的特性。

6)枚举(号称最好)

  1. public enum EnumSingleton01 {
  2. INSTANCE;
  3. public void doSomething() {
  4. System.out.println("doSomething");
  5. }
  6. }

模拟数据库链接:

  1. public enum EnumSingleton02 {
  2. INSTANCE;
  3. private DBConnection dbConnection = null;
  4. private EnumSingleton02() {
  5. dbConnection = new DBConnection();
  6. }
  7. public DBConnection getConnection() {
  8. return dbConnection;
  9. }
  10. }

这种方式是Effective Java作者Josh Bloch提倡的方式,它不仅能避免多线程同步问题,
而且还能防止反序列化重新创建新的对象。

2、为什么说枚举方法是最好的?

前5种方式实现单例都有如下3个特点:

  • 构造方法私有化
  • 实例化的变量引用私有化
  • 获取实例的方法共有

首先,私有化构造器并不保险。因为它抵御不了反射攻击,其次就是序列化重新创建新对象。下面来进行验证。

1) 反射验证

  1. @Test
  2. public void reflectTest() throws Exception {
  3. Singleton03 s = Singleton03.getInstance();
  4. // 拿到所有的构造函数,包括非public的
  5. Constructor<Singleton03> constructor = Singleton03.class.getDeclaredConstructor();
  6. constructor.setAccessible(true);
  7. // 构造实例
  8. Singleton03 reflection = constructor.newInstance();
  9. System.out.println(s);
  10. System.out.println(reflection);
  11. System.out.println(s == reflection);
  12. }

输出结果:

  1. org.yd.singleton.Singleton03@61e4705b
  2. org.yd.singleton.Singleton03@50134894
  3. false

再看看枚举类的测试

  1. @Test
  2. public void reflectEnumTest() throws Exception {
  3. EnumSingleton01 s = EnumSingleton01.INSTANCE;
  4. // 拿到所有的构造函数,包括非public的
  5. Constructor<EnumSingleton01> constructor = EnumSingleton01.class.getDeclaredConstructor();
  6. constructor.setAccessible(true);
  7. // 构造实例
  8. EnumSingleton01 reflection = constructor.newInstance();
  9. System.out.println(s);
  10. System.out.println(reflection);
  11. System.out.println(s == reflection);
  12. }

输出结果:

  1. java.lang.NoSuchMethodException: org.yd.singleton.EnumSingleton01.<init>()
  2. at java.lang.Class.getConstructor0(Class.java:3082)
  3. at java.lang.Class.getDeclaredConstructor(Class.java:2178)
  4. at org.yd.singleton.SingletonTest.reflectEnumTest(SingletonTest.java:61)
  5. at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
  6. at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
  7. at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)

结论:通过反射,单例模式的私有构造方法也能构造出新对象。不安全。而枚举类直接抛异常,说明枚举类对反射是安全的。

2) 序列化验证

  1. @Test
  2. public void serializeTest(){
  3. Singleton03 s = Singleton03.getInstance();
  4. String serialize = JSON.toJSONString(s);
  5. Singleton03 deserialize =JSON.parseObject(serialize,Singleton03.class);
  6. System.out.println(s);
  7. System.out.println(deserialize);
  8. System.out.println(s == deserialize);
  9. }

输出结果:

  1. org.yd.singleton.Singleton03@387c703b
  2. org.yd.singleton.Singleton03@75412c2f
  3. false

结论:序列化前后两个对象并不相等。所以序列化也是不安全的。

同样看看枚举类的测试

  1. @Test
  2. public void serializeEnumTest(){
  3. EnumSingleton01 s = EnumSingleton01.INSTANCE;
  4. String serialize = JSON.toJSONString(s);
  5. EnumSingleton01 deserialize =JSON.parseObject(serialize,EnumSingleton01.class);
  6. System.out.println(s);
  7. System.out.println(deserialize);
  8. System.out.println(s == deserialize);
  9. }

输出结果:

  1. INSTANCE
  2. INSTANCE
  3. true

结论:说明枚举类序列化安全。


综上,可以得出结论:枚举是实现单例模式的最佳实践。

  • 反射安全
  • 序列化/反序列化安全
  • 写法简单

原文链接:http://www.cnblogs.com/zs-yuandian/p/14633829.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号