一、简介

  单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

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

二、主要特征

  1.单例类只能有一个实例(即单例类不能被外部实例化---单例类的构造方法由private修饰)。

  2.单例类必须自己创建自己的实例,并且供外部调用(向外部暴露调用其实例的静态方法)。

三、优缺点

  1.优点:

    1)在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例。

    2)不需要去重复的创建实例,避免对资源的多重占用(比如写文件操作)。

  2.缺点:

    没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。

四、实现

  1.场景:一个班级有很多学生,但只有一个班主任

  2.代码实现:

    1)饿汉式:即类加载时就实例化,相对浪费内存,并且容易产生垃圾对象

public class Teacher {
/**
* 创建自己的实例
*/
private static Teacher teacher = new Teacher(); /**
* 静态构造方法,不允许外部对其实例化
*/
private Teacher(){} /**
* 将自己的实例对象,提供给外部使用
* @return Teacher
*/
public static Teacher getInstance(){
return teacher;
} public void attendClass(){
System.out.println("班主任说:上课");
} public void finishClass(){
System.out.println("班主任说:下课");
}
}

    2)懒汉式:第一次调用才实例化,避免浪费内存

public class Teacher {
/**
* 创建自己的实例
*/
private static Teacher teacher; /**
* 静态构造方法,不允许外部对其实例化
*/
private Teacher(){} /**
* 将自己的实例对象,提供给外部使用
* @return Teacher
*/
public static Teacher getInstance(){
     if(null == teacher){
       teacher = new Teacher();
     }
return teacher;
} public void attendClass(){
System.out.println("班主任说:上课");
} public void finishClass(){
System.out.println("班主任说:下课");
}
}

注:懒汉式加载是否为线程安全的区别在于synchronized 关键字,给getInstance()方法加上synchronized 关键字,使得线程安全,能够在多线程中很好的工作,但是,效率很低,99% 情况下不需要同步。

    3)双检锁/双重校验锁(DCL,即 double-checked locking):采用双锁机制,安全且在多线程情况下能保持高性能。getInstance() 的性能对应用程序很关键。

public class Teacher {
/**
* 创建自己的实例
*/
private volatile static Teacher teacher = new Teacher(); /**
* 静态构造方法,不允许外部对其实例化
*/
private Teacher(){} /**
* 将自己的实例对象,提供给外部使用
* @return Teacher
*/
public static Teacher getInstance(){
if (teacher == null) {
synchronized (Teacher.class) {
if (teacher == null) {
teacher = new Teacher();
}
}
}
return teacher;
} public void attendClass(){
System.out.println("班主任说:上课");
} public void finishClass(){
System.out.println("班主任说:下课");
}
}

    4)登记式/静态内部类:与双检索方式功效一样,但实现更简单;这种方式同样在类加载时不会实例化,只有在外部调用其getInstance()方法时才会被实例化

public class Teacher {
/**
* 静态内部类
*/
private static class TeacherHolder {
/**
* 创建Teacher实例,只有在显示的调用时才会被实例化
*/
private static final Teacher INSTANCE = new Teacher();
}
/**
* 静态构造方法,不允许外部对其实例化
*/
private Teacher (){} /**
* 将单例类,提供给外部使用
* @return Teacher
*/
public static final Teacher getInstance() {
return TeacherHolder.INSTANCE;
} public void attendClass(){
System.out.println("班主任说:上课");
} public void finishClass(){
System.out.println("班主任说:下课");
}
}

    5)枚举:实现单例模式的最佳方法,更简洁,自动支持序列化机制,绝对防止多次实例化(目前未被广泛使用)

public enum Teacher {
INSTANCE;
public void whateverMethod() {
System.out.println("枚举下单例模式");
}
}

以上代码讲述了如何创建一个单例类Teacher,下面来看一下如何使用这个单例

public class Test {
public static void main(String[] args) {
//调用单例类供外部调用的方法,获取其实例
Teacher teacher = Teacher.getInstance();
teacher.attendClass();
teacher.finishClass();
}
}

这样一个简单的单例模式的例子就已经完成了,接下来举一个实际应用的例子,例如:jdbc

public class DBUtils {
public static final String URL = "jdbc:mysql://localhost:3306/demo";
public static final String USER = "root";
public static final String PASSWORD = "root";
private static Connection conn = null;
static{
try {
//1.加载驱动程序
Class.forName("com.mysql.jdbc.Driver");
//2. 获得数据库连接
conn = DriverManager.getConnection(URL, USER, PASSWORD);
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (SQLException e) {
e.printStackTrace();
}
} public static Connection getConnection(){
return conn;
}
}
public class TeacherDB {
private int id;
private String name; public int getId() {
return id;
} public void setId(int id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} @Override
public String toString() {
return "TeacherDB{" +
"id=" + id +
", name='" + name + '\'' +
'}';
}
}
public class TestDB {
public static boolean addTeacher() {
//获取jdbc连接
Connection connection = DBUtils.getConnection();
//sql
String sql = "INSERT INTO teacher(name) values(?)";
PreparedStatement ptmt = null;
try {
//预编译SQL,减少sql执行
ptmt = connection.prepareStatement(sql);
ptmt.setString(1,"john");
return ptmt.execute();
} catch (SQLException e) {
e.printStackTrace();
}
return false;
} public static List<TeacherDB> queryTeacher() {
//获取jdbc连接
Connection connection = DBUtils.getConnection();
//sql
String sql = "select * from teacher";
Statement stmt = null;
ResultSet rs = null;
List<TeacherDB> list = new ArrayList<>();
try {
stmt = connection.createStatement();
rs = stmt.executeQuery(sql);
while (rs.next()){
TeacherDB teacherDB = new TeacherDB();
teacherDB.setId(rs.getInt("id"));
teacherDB.setName(rs.getString("name"));
list.add(teacherDB);
}
return list;
} catch (SQLException e) {
e.printStackTrace();
}
return list;
} public static void main(String[] args) {
System.out.println(addTeacher());
List<TeacherDB> list = queryTeacher();
list.stream().forEach(item->{
System.out.println(item);
});
}
}

五、总结

  一般情况下,不建议使用懒汉方式,建议使用饿汉方式。只有在要明确实现 lazy loading 效果时,才会使用登记式/静态内部类的方式。如果涉及到反序列化创建对象时,可以尝试使用枚举方式。如果有其他特殊的需求,可以考虑使用双检锁/双重校验锁的方式。本篇就到此结束了,谢谢观看

《java常用设计模式之----单例模式》的更多相关文章

  1. 简单物联网:外网访问内网路由器下树莓派Flask服务器

    最近做一个小东西,大概过程就是想在教室,宿舍控制实验室的一些设备. 已经在树莓上搭了一个轻量的flask服务器,在实验室的路由器下,任何设备都是可以访问的:但是有一些限制条件,比如我想在宿舍控制我种花 ...

  2. 利用ssh反向代理以及autossh实现从外网连接内网服务器

    前言 最近遇到这样一个问题,我在实验室架设了一台服务器,给师弟或者小伙伴练习Linux用,然后平时在实验室这边直接连接是没有问题的,都是内网嘛.但是回到宿舍问题出来了,使用校园网的童鞋还是能连接上,使 ...

  3. 外网访问内网Docker容器

    外网访问内网Docker容器 本地安装了Docker容器,只能在局域网内访问,怎样从外网也能访问本地Docker容器? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Docker容器 ...

  4. 外网访问内网SpringBoot

    外网访问内网SpringBoot 本地安装了SpringBoot,只能在局域网内访问,怎样从外网也能访问本地SpringBoot? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装Java 1 ...

  5. 外网访问内网Elasticsearch WEB

    外网访问内网Elasticsearch WEB 本地安装了Elasticsearch,只能在局域网内访问其WEB,怎样从外网也能访问本地Elasticsearch? 本文将介绍具体的实现步骤. 1. ...

  6. 怎样从外网访问内网Rails

    外网访问内网Rails 本地安装了Rails,只能在局域网内访问,怎样从外网也能访问本地Rails? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Rails 默认安装的Rails端口 ...

  7. 怎样从外网访问内网Memcached数据库

    外网访问内网Memcached数据库 本地安装了Memcached数据库,只能在局域网内访问,怎样从外网也能访问本地Memcached数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装 ...

  8. 怎样从外网访问内网CouchDB数据库

    外网访问内网CouchDB数据库 本地安装了CouchDB数据库,只能在局域网内访问,怎样从外网也能访问本地CouchDB数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Cou ...

  9. 怎样从外网访问内网DB2数据库

    外网访问内网DB2数据库 本地安装了DB2数据库,只能在局域网内访问,怎样从外网也能访问本地DB2数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动DB2数据库 默认安装的DB2 ...

  10. 怎样从外网访问内网OpenLDAP数据库

    外网访问内网OpenLDAP数据库 本地安装了OpenLDAP数据库,只能在局域网内访问,怎样从外网也能访问本地OpenLDAP数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动 ...

随机推荐

  1. Java Service Wrapper 浅谈

    在实际开发过程中很多模块需要独立运行,他们并不会以web形式发布,传统的做法是将其压缩为jar包独立运行,这种形式简单易行也比较利于维护,但是 一旦服务器重启或出现异常时,程序往往无法自行修复或重启. ...

  2. 下订单更新订单表然后减少库存表中的数据,出现库存超卖,使用数据库和redis坚决库存超卖的问题

    上面的代码更新库存的数据,存在多线程的问题,第一种方法使用synchronized关键字修饰的语句块代码,但是性能较低,并且还是存在问题的 在分布式的场景下,当前库存系统部署在多个tomcat上,即使 ...

  3. 深入理解React:事件机制原理

    目录 序言 DOM事件流 事件捕获阶段.处于目标阶段.事件冒泡阶段 addEventListener 方法 React 事件概述 事件注册 document 上注册 回调函数存储 事件分发 小结 参考 ...

  4. nova api报错network问题

    安装openstack Rocky版本的时候,在未安装网络服务前,创建虚拟机,报以下错误 [root@controller2 nova]# openstack server create --flav ...

  5. 路由网关(Zuul)

    上一篇已经讲了微服务组件中的 分布式配置中心,本章讲述 由JAVA编写的服务路由网关Zuul… - Zuul 路由是微服务体系结构的一个组成部分.例如 / 可以映射到您的Web应用程序,/api/us ...

  6. 强大的 actuator 服务监控与管理

    SpringBoot 是为了简化 Spring 应用的创建.运行.调试.部署等一系列问题而诞生的产物,自动装配的特性让我们可以更好的关注业务本身而不是外部的XML配置,我们只需遵循规范,引入相关的依赖 ...

  7. C#模拟HTTP协议请求

    1       前言 http协议详解参考HTTP协议. WEB常用Get和Post请求. 地址栏参数:url为www.viwofer.com?id=12&name=viwofer则id=12 ...

  8. python抓取头条文章

    python抓取头条美文并存储到mongodb # Author:song from multiprocessing import Pool from urllib.parse import urle ...

  9. 每日一题 - 剑指 Offer 43. 1~n整数中1出现的次数

    题目信息 时间: 2019-07-01 题目链接:Leetcode tag: 整除 取余 规律 递归 难易程度:中等 题目描述: 输入一个整数 n ,求1-n这n个整数的十进制表示中1出现的次数. 例 ...

  10. 机器分配——线性dp输出路径

    题目描述 总公司拥有高效设备M台, 准备分给下属的N个分公司.各分公司若获得这些设备,可以为国家提供一定的盈利.问:如何分配这M台设备才能使国家得到的盈利最大?求出最大盈利值.其中M <= 15 ...