Java单例模式简单介绍
一、概念
单例模式是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例类的特殊类。通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。就笔者认为,单例就是不让外界创建对象。
1.1概念剖析
对于单例的话,从上面的概念剖析,应该满足下面的几个条件:
第一:单例类中只能有一个单例对象;
第二:单例类必须自己创建自己的唯一实例对象;
第三:这个实例对象能够给外界访问到,并且外界不能够自己创建对象。
二、常见几种单例模式的方式
在java中,对于单例模式一般来说,分为懒汉式,饿汉式,以及登记式,但是登记式一般较少看到,所以也容易忽略。笔者若非今天突然想总结一下,在网上查找资料,也不会注意到这个。下面按照这种方式来贴出代码,并进行解释。
2.1饿汉式单例类
package com.ygh.singleton; /** * 饿汉式单例类 * @author 夜孤寒 * @version 1.1.1 */ public class HungerSingleton { //将构造方法私有,外界类不能使用构造方法new对象 private HungerSingleton(){} //创建一个对象 private static final HungerSingleton lazySinleton=new HungerSingleton(); //设置实例获取方法,返回实例给调用者 public static HungerSingleton getInstance(){ return lazySinleton; } }
写一个测试类,测试是不是实现单例:
package com.ygh.singleton; /** * 测试单例类 * * @author 夜孤寒 * @version 1.1.1 */ public class Test { public static void main(String[] args) { /* * 构造方法私有化,不能够使用下面方式new对象 */ //HungerSingleton hungerSingleton=new HungerSingleton(); //使用实例获取方法来获取对象 HungerSingleton h1=HungerSingleton.getInstance(); HungerSingleton h2=HungerSingleton.getInstance(); System.out.println(h1==h2);//true } }
从上面可以看出:这个测试类的两个引用是相等的,也就是说两个引用指向的是同一个对象,而这也正好符合单例模式标准。到这里,饿汉式介绍结束。
2.2饿汉式单例类
package com.ygh.singleton; /** * 懒汉式单例类 * @author 夜孤寒 * @version 1.1.1 */ public class LazySingleton { //将构造方法私有,外界类不能使用构造方法new对象 private LazySingleton(){} //创建一个对象,不为final private static LazySingleton lazySingleton=null; //设置实例获取方法,返回实例给调用者 public static LazySingleton getInstance(){ //当单例对象不存在,创建 if(lazySingleton==null){ lazySingleton=new LazySingleton(); } //返回 return lazySingleton; } }
测试类:
package com.ygh.singleton; /** * 测试单例类 * * @author 夜孤寒 * @version 1.1.1 */ public class Test { public static void main(String[] args) { /* * 构造方法私有化,不能够使用下面方式new对象 */ //LazySingleton lazySingleton=new LazySingleton(); //使用实例获取方法来获取对象 LazySingleton l1=LazySingleton.getInstance(); LazySingleton l2=LazySingleton.getInstance(); System.out.println(l1==l2);//true } }
从上面可以看出:这个测试类的两个引用是相等的,也就是说两个引用指向的是同一个对象,而这也正好符合单例模式标准。到这里,懒汉式介绍结束。
2.3懒汉式和饿汉式的区别
懒汉式是当没有对象的时候,就会创建一个单例对象,当有对象的时候,就不会再创建对象,这个说起来可能不是那么容易理解,但是读者如果有兴趣了解更深,可以在eclipse中使用断点来测试,将LazySingleton类的if花括号内的内容加上断点,然后在Test类中,使用debug运行,这样子就能够很容易体现出来,第一次创建了一个对象,但是第二次没有创建对象。
饿汉式则是实现就用final这个关键字将对象创建好了,当调用者需要实例对象的时候,就可以通过getInstance这个方法获取创建好的实例。
2.4登记式单例类
对于登记式单例类,笔者也不是很熟悉,贴了一段网络上的代码以供自己学习参考,请读者自行学习。
import java.util.HashMap; import java.util.Map; /** * 登记式单例类 * @author Administrator * */ public class RegisterSingleton { private static Map<String, RegisterSingleton> map = new HashMap<String, RegisterSingleton>(); static { RegisterSingleton single = new RegisterSingleton(); map.put(single.getClass().getName(), single); } /* * 保护的默认构造方法 */ protected RegisterSingleton() { } /* * 静态工厂方法,返还此类惟一的实例 */ public static RegisterSingleton getInstance(String name) { if (name == null) { name = RegisterSingleton.class.getName(); System.out.println("name == null" + "--->name=" + name); } if (map.get(name) == null) { try { map.put(name, (RegisterSingleton) Class.forName(name).newInstance()); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } } return map.get(name); } /* * 一个示意性的商业方法 */ public String about() { return "Hello, I am RegSingleton."; } public static void main(String[] args) { RegisterSingleton single3 = RegisterSingleton.getInstance(null); System.out.println(single3.about()); } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
- 简单的实现Javascript的MVC
- 谷歌再出黑科技!人工智能模拟的人声和真人几乎难以分辨
- 【Scikit-Learn 中文文档】高斯混合模型 - 无监督学习 - 用户指南 | ApacheCN
- “E+单拼”组合域名eqi.com小六位结拍
- ASP.NET Core中的依赖注入(3): 服务的注册与提供
- 【Scikit-Learn 中文文档】使用 scikit-learn 介绍机器学习 | ApacheCN
- 哈萨比斯:2018年AI最大突破在生物或化学领域
- ASP.NET Core中的依赖注入(5): ServiceProvider实现揭秘 【总体设计 】
- ASP.NET MVC的View是如何被呈现出来的?[设计篇]
- Python大牛告诉你一行代码能干什么?神奇
- 谷歌最新人工智能研究:仅利用稀疏轮廓位置“重构”图像
- DATUM和BigchainDB
- MySQL主从复制的实现过程
- 使用dict和set
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- 第05期:Prometheus 数据查询(一)
- 技术分享 | MySQL 复制那点事 - Seconds_behind_Master 参数调查笔记
- 线程有多少种状态?Runnable 一定在执行任务吗?
- swift 中类(class)和结构体(struct)区别
- C语言三剑客之《C专家编程》一书精华提炼
- 前端必备技能:json-server全攻略
- 温故知新——Spring AOP(二)
- R语言中的偏最小二乘PLS回归算法
- R如何与Tableau集成分步指南
- 漏洞复现 | CVE‐2020‐5902踩坑记
- PCA图显示分组无差异,怎么办?
- 我接手了一个“垃圾”系统,全栈优化后将性能提升了350倍
- 用 eggnog-mapper 进行功能注释
- 将SAP C4C Custom BO使用ABSL编写的逻辑通过OData服务暴露出去
- Linux笔记(14)| 字符设备驱动基础入门(续)