在平时的工作、学员的学习以及面试过程中,单例模式作为一种常用的设计模式,会经常被面试官问到,甚至笔试会要求学员现场默写,下面将会就单例模式的实现思路和几种常见的实现方式进行简单的分享。

单例模式,是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中,应用该模式的类一个类只有一个实例。即一个类只有一个对象实例。是最常用到的设计模式之一,熟悉设计模式的朋友对单例模式都不会陌生。一般介绍单例模式的书籍都会提到 饿汉式 和 懒汉式 这两种实现方式。但是除了这两种方式,本文还会介绍其他几种实现单例的方式。

基本的实现思路

单例模式要求类能够有返回对象一个引用(永远是同一个)和一个获得该实例的方法(必须是静态方法,通常使用getInstance这个名称)。

通俗的讲:即设计一个类,在整个应用中只存在一个对象(需要做到让本类的外部不能够随意创建对象)

单例的实现主要是通过以下个步骤:

① 构造方法私有化(暂时不考虑反射)

② 在内部创建好一个对象并保存起来

③ 向外提供一个公共的静态的方法,返回内部对象的地址

第一种实现方式:饿汉式[静态常量]的方式

示例代码:

package cn.itsource.sington;

/**

 * 饿汉式[静态常量]

*

* @author Administrator

*

*/

public class SingletonTest1 {

/**

* 单例模式 设计一个类 ,让这个 类中只存在一个对象的实例

*

* 1 需要一个私有的 构造方法,这样可以避免外部创建对象

*/

private SingletonTest1() {

}

/*

* 2 在本类内部创建一个 对象 ,保存起来

*/

private static SingletonTest1 instance = new SingletonTest1();

/*

* 3 向外公布一个 公共的 静态的方法 ,返回内部保存的这个对象

*

*/

public static SingletonTest1 getInstance() {

return instance;

}

}

优点:这种写法比较简单,就是在类装载的时候就完成实例化。避免了线程同步问题。

缺点:对象的创建是类加载的时候,可能会导致类加载很慢,没有达到Lazy Loading的效果。如果从始至终从未使用过这个实例,则会造成内存的浪费。

第二种实现方式:饿汉式[静态代码块]的方式

示例代码:

package cn.itsource.sington;

public class SingletonTest3 {

private SingletonTest3() {

}

private static SingletonTest3 instance;

static {

instance = new SingletonTest3();

}

public static SingletonTest3 getInstance() {

return instance;

}

}

这种方式和上面的方式其实类似,只不过将类实例化的过程放在了静态代码块中,也是在类装载的时候,就执行静态代码块中的代码,初始化类的实例。优缺点和上面是一样的。

第三种实现方式:懒汉式[双重检查]的方式

示例代码:

package cn.itsource.sington;

/**

* 懒汉式

* @author Administrator

*/

public class SingletonTest2 {

/**

* 需求:单例模式 设计一个类 ,让这个 类中只存在一个对象的实例

*

* 1 需要一个私有的 构造方法,这样可以避免外部创建对象

*/

private SingletonTest2() {

}

/*

* 2 类加载进内存的时候,对象还没有存在,只有调用了getInstance()方法时,对

象才开始创建

*/

private static SingletonTest2 instance;

/*

* 3 向外公布一个 公共的 静态的方法 ,返回内部保存的这个对象

*   懒汉式是延迟加载,如果多个线程同时操作懒汉式时就有可能出现线程安全问题,

解决线程安全问题:

可以加同步来解决。但是加了同步之后,每一次都要比较锁,效率就变慢了,

所以可以加双重判断来提高程序效率。

*/

public static SingletonTest2 getInstance() {

if (instance == null) {

synchronized (SingletonTest2.class) {

if (instance == null) {

instance = new SingletonTest2();

}

}

}

return instance;

}

}

双重判断,对于多线程开发者来说不会陌生,如代码中所示,我们进行了两次if (instance == null)检查,这样就可以保证线程安全了。这样,实例化代码只用执行一次,后面再次访问时,判断if (instance == null),直接return实例化对象。

优点:线程安全;延迟加载;效率较高。

饿汉式和懒汉式的区别:

1饿汉式是类一加载进内存就创建好了对象;

2懒汉式则是类加载进内存的时候,对象还没有存在,只有调用了getInstance()方法时,对象才开始创建。

3懒汉式是延迟加载,如果多个线程同时操作懒汉式时就有可能出现线程安全问题,解决线程安全问题可以加同步来解决。但是加了同步之后,每一次都要比较锁,效率就变慢了,所以可以加双重判断来提高程序效率。

第四种实现方式:枚举的方式

示例代码:

package cn.itsource.sington;

public enum SingletonTest4 {

INSTANCE;

}

借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。可能是因为枚举在JDK1.5中才添加,所以在实际项目开发中,很少见人这么写过。

优点写法简单这是它最大的优点,其次可以自己处理序列化,是线程安全的

缺点当想实例化一个单例类的时候,必须要记住使用SingletonTest4.INSTANCE获取对象的方法,而不是使用new,可能会给其他开发人员造成困扰,特别是看不到源码的时候。

以上是对单例模式常见的几种实现方式,在教学和学习过程中的一点简单的总结,希望对大家学习单例模式有一点点帮助。

Java单例模式几种实现方式的更多相关文章

  1. Java 8 的新特性和Java 的4种引用方式

    一.接口的增强 Java 8允许我们给接口添加一个非抽象的方法实现,只需要使用 default关键字即可,这个特征又叫做扩展方法,示例如下: interface Formula { double ca ...

  2. java的两种同步方式, Synchronized与ReentrantLock的区别

    java在编写多线程程序时,为了保证线程安全,需要对数据同步,经常用到两种同步方式就是Synchronized和重入锁ReentrantLock. 相似点: 这两种同步方式有很多相似之处,它们都是加锁 ...

  3. java集合四种遍历方式

    package conection; import java.util.Iterator;import java.util.LinkedList;import java.util.List; publ ...

  4. Java多线程--两种实现方式

    进程概述: 在这之前,有必要了解一下什么是进程? 在一个操作系统中,每个独立的执行的程序都可称为一个进程,也就是"正在运行的程序".如图所示: 线程概述: 如上所述,每个运行的程序 ...

  5. 转载:Java的四种引用方式

    原文:https://www.cnblogs.com/huajiezh/p/5835618.html Java内存管理分为内存分配和内存回收,都不需要程序员负责,垃圾回收的机制主要是看对象是否有引用指 ...

  6. Java HashMap两种遍历方式

    第一种: Map map = new HashMap(); Iterator iter = map.entrySet().iterator(); while (iter.hasNext()) { Ma ...

  7. JAVA多线程三种实现方式

    JAVA多线程实现方式主要有三种:继承Thread类.实现Runnable接口.使用ExecutorService.Callable.Future实现有返回结果的多线程.其中前两种方式线程执行完后都没 ...

  8. Java 多线程 三种实现方式

    Java多线程实现方式主要有三种:继承Thread类.实现Runnable接口.使用ExecutorService.Callable.Future实现有返回结果的多线程.其中前两种方式线程执行完后都没 ...

  9. 在java 中一种简单方式的声明静态Map常量的方法

    我现在需要在一个类里面放一个HashMap,往里面放一些数据,每次要从数据库中取数据的时候先查找HashMap,看是否已经存在,若存在就直接提取,若不存在就从数据库中抽取数据之后再放到HashMap中 ...

随机推荐

  1. Android踩坑随笔Fragment中onActivityResult方法不被调用

    最近项目里要做头像功能,参考了这篇博客(GitHub - zhudfly/SelectAvatarApplication: 一个选择并显示头像圆形控件,可以通过拍照或者选择相册中的图片来设置图片),但 ...

  2. 媒体查询hack

    随着Responsive设计的流行,Medial Queries可算是越来越让人观注了.他可以让Web前端工程实现不同设备下的样式选择,让站点在不同的设备中实现不同的效果.这个早前在 w3cplus已 ...

  3. MySQL允许root远程访问

    1. mysql -u root -p;     // 登录mysql, 并输入密码 2. use mysql;             // 打开 mysql 数据库 3. update user ...

  4. chchc

    ---恢复内容开始--- 51CTO博客-原创IT文章分享平台 Logo 首页 文章 专家 专家博客 博客之星 推荐博客 我的博客 网站导航 学院 博客 下载 家园 论坛 CTO训练营 WOT 51C ...

  5. Linux bash内置命令集

    man cd  -->查询不到,所以会提示bash的内置命令 . alias bg bind break builtin caller cd command compgen complete c ...

  6. 深入浅出SharePoint2010——请假系统无代码篇之数据框架设计

    文档库SOP:上传用户操作手册等系统相关文档. 员工信息列表EmployeeInfo:用来存储员工基本信息.直属领导和假卡信息. 请假申请列表LeaveRequest:申请人Requester填写请假 ...

  7. August 16th 2017 Week 33rd Wednesday

    A man can be destroyed but not defeated. 一个人可以被毁灭,但不能被打败. Before he was destroyed, he would have bee ...

  8. Linux命令--网络管理

    write命令 Linux write命令用于传讯息给其他使用者. 使用权限:所有使用者. 语法 write user [ttyname] 参数说明: user : 预备传讯息的使用者帐号 ttyna ...

  9. Redis数据的底层存储原理

    redis底层是用什么结构来存储数据的呢? 我们从源码上去理解就会容易的多:   redis底层是使用C语言来编写的,我们可以看到它的数据结构声明.一个 dict 有两个dictht,一个dictht ...

  10. ClassNotFoundException: INameEnvironment

    springboot兼容jsp启动是报错,ClassNotFoundException: INameEnvironment 降低springboot版本即可,原版本1.5.10,降为1.5.2可使用