单例模式在实际开发中有很多的用途,比如我们在项目中常用的工具类,数据库等资源的连接类。这样做的好处是避免创建多个对象,占用内存资源,自始自终在内存中只有一个对象为我们服务。

单例对象一般有两种实现方式。懒汉式和饿汉式。

饿汉式实现如下:

package com.day05;

/**
* 饿汉式
*
* @author Administrator
*
*/
public class Single {
// 定义一个个私有静态本类对象变量,饿汉式是一上来就给该变量进行初始化,加上final是让s这个对象引用始终保持不变,更为严谨
private static final Single s = new Single(); // 构造方法私有化,让外部无法通过new 关键字来进行创建对象
private Single() {
} // 暴露出一个外界可以获取该类对象的公共静态方法
public static Single getInstance() {
return s;
} }

测试类:

package com.day05;

public class SingleDemo {
public static void main(String[] args) {
Single s1 = Single.getInstance();
Single s2 = Single.getInstance();
System.out.println(s1==s2); }
}

运行结果:

true
即s1==s2说明了,s1和s2在内存中地址都相等,即s1、和s2是同一个对象。

懒汉式实现如下:

package com.day05;

/**
* 懒汉式
*
* @author Administrator
*
*/
public class Single {
// 定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化
private static Single s = null; // 构造方法私有化,让外部无法通过new 关键字来进行创建对象
private Single() {
} // 暴露出一个外界可以获取该类对象的公共静态方法
public static Single getInstance() {
if (s == null)
s = new Single();
return s;
} }

以上的代码如果是单线程的话就不会存在问题,但是当有多线程操作的时候,就会存在线程安全问题,演示代码如下:

package com.day05;

/**
* 懒汉式
*
* @author Administrator
*
*/
public class Single {
// 定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化
private static Single s = null; // 构造方法私有化,让外部无法通过new 关键字来进行创建对象
private Single() {
} // 暴露出一个外界可以获取该类对象的公共静态方法
public static Single getInstance() {
if (s == null) {
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
s = new Single();
}
return s;
} }

测试类如下:

package com.day05;

public class SingleDemo implements Runnable {

    public static void main(String[] args) {
SingleDemo sd = new SingleDemo();
new Thread(sd).start();
new Thread(sd).start(); } @Override
public void run() {
Single s = Single.getInstance();
System.out.println(s); }
}

运行结果如下:

com.day05.Single@4081b5a4
com.day05.Single@64dcdaac

可以发现每次运行结果打印出获取对象不是同一个,即存在线程安全问题。

问题分析:

由此我们可以采用Java给我们提供的同步锁来解决以上的问题,修改代码如下:

package com.day05;

/**
* 懒汉式
*
* @author Administrator
*
*/
public class Single {
// 定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化
private static Single s = null; // 构造方法私有化,让外部无法通过new 关键字来进行创建对象
private Single() {
} // 暴露出一个外界可以获取该类对象的公共静态方法
public static synchronized Single getInstance() {
if (s == null) {
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
s = new Single();
}
return s;
} }

这样就解决了上面的代码存在的线程安全问题,但是同步锁虽然可以解决了线程安全问题,但是却会存在效率问题,所以我们可以采用双重判断的方法来优化一下改代码如下所示:

package com.day05;

/**
* 懒汉式
*
* @author Administrator
*
*/
public class Single {
// 定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化
private static Single s = null; // 构造方法私有化,让外部无法通过new 关键字来进行创建对象
private Single() {
} // 暴露出一个外界可以获取该类对象的公共静态方法
// 这里需要注意的使静态共享数据使用的使该类的字节码对象即Single.class
public static Single getInstance() {
// 这里增加了一次判断,可以少一次进行锁的处理
if (s == null) {
synchronized (Single.class) {
if (s == null) {
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
s = new Single();
}
}
} return s;
} }

总结:还是比较推荐使用饿汉式,因为写法简单,不存在线程安全问题。

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

  1. 对Java代码加密的两种方式,防止反编译

    使用Virbox Protector对Java项目加密有两种方式,一种是对War包加密,一种是对Jar包加密.Virbox Protector支持这两种文件格式加密,可以加密用于解析class文件的j ...

  2. Java新建线程的两种方式

    Java新建线程有两种方式,一种是通过继承Thread类,一种是实现Runnable接口,下面是新建线程的两种方式. 我们假设有个竞赛,有一个选手A做俯卧撑,一个选手B做仰卧起坐.分别为两个线程: p ...

  3. Java实现多线程的两种方式

    实现多线程的两种方式: 方式1: 继承Thread类 A: 自定义MyThread类继承Thread类 B: 在MyThread类中重写run() C: 创建MyThread类的对象 D: 启动线程对 ...

  4. [Java] HashMap遍历的两种方式

    Java中HashMap遍历的两种方式原文地址: http://www.javaweb.cc/language/java/032291.shtml第一种: Map map = new HashMap( ...

  5. Java实现深克隆的两种方式

    序列化和依次克隆各个可变的引用类型都可以实现深克隆,但是序列化的效率并不理想 下面是两种实现深克隆的实例,并且测试类对两种方法进行了对比: 1.重写clone方法使用父类中的clone()方法实现深克 ...

  6. java文件读写的两种方式

    今天搞了下java文件的读写,自己也总结了一下,但是不全,只有两种方式,先直接看代码: public static void main(String[] args) throws IOExceptio ...

  7. K:java中序列化的两种方式—Serializable或Externalizable

    在java中,对一个对象进行序列化操作,其有如下两种方式: 第一种: 通过实现java.io.Serializable接口,该接口是一个标志接口,其没有任何抽象方法需要进行重写,实现了Serializ ...

  8. java 实现websocket的两种方式

    简单说明 1.两种方式,一种使用tomcat的websocket实现,一种使用spring的websocket 2.tomcat的方式需要tomcat 7.x,JEE7的支持. 3.spring与we ...

  9. java实现同步的两种方式

    同步是多线程中的重要概念.同步的使用可以保证在多线程运行的环境中,程序不会产生设计之外的错误结果.同步的实现方式有两种,同步方法和同步块,这两种方式都要用到synchronized关键字. 给一个方法 ...

随机推荐

  1. 【Android Studio探索之路系列】之中的一个:Android Studio开篇

    作者:郭孝星 微博:郭孝星的新浪微博 邮箱:allenwells@163.com 博客:http://blog.csdn.net/allenwells github:https://github.co ...

  2. 【Python】self的用法扫盲

    在Python中,我们有两个重要的概念:类与实例 例如:我们在现实生活中人就是一个类,实例就是具体到某一个男人(张三.李四等) 1.类:定义人这个类 class People(object): pas ...

  3. [linux]w命令和uptime命令查看系统负载

    在Linux系统中查询系统CPU和内存的负载(使用率)时,我们通常习惯于使用top.atop或者ps,这篇文章将要给大家介绍如何使用w命令和uptime命令来查看系统的负载情况,对于uptime命令, ...

  4. 【转】前端上传组件Plupload

    [转自博客园-无双] html5原生的给我们提供了文件上传的API,Plupload是一款由著名的web编辑器TinyMCE团队开发的上传组件,简单易用且功能强大,我们完全可以使用Plupload来代 ...

  5. CentOS 7 中 Docker 的安装和卸载

    安装Dokcer Docker 软件包已经包括在默认的 CentOS-Extras 软件源里.因此想要安装 docker,只需要运行下面的 yum 命令: [root@localhost ~]# yu ...

  6. 解析式/推导式, 生成器 datetime 内建函数

    列表解析式(List Comprehension) 语法: [返回值 for 元素 in 可迭代对象 if 条件] 使用中括号[],内部是for循环,if条件可选. 返回一个新的列表. 列表解析式的作 ...

  7. 未能加载文件或程序集“WebGrease, Version=1.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35”或它的某一个依赖项。找到的程序集清单定义与程序集引用不匹配。 (异常来自 HRESULT:0x80131040)

    方法一:在web.config的configuration接点中添加,最好是添加在configuration节点的最后 <runtime> <assemblyBinding xmln ...

  8. 如何给unity3d工程加入依赖的android工程

    最近在忙着接平台的事,需要接入各种各样的android平台sdk来发布.在接sdk的时候遇到了这样的一个情况,有点麻烦,所以纪录一下. 有些sdk的接入是提供jar包,这样的可以简单地将jar包制作成 ...

  9. Android 之布局

    1.RelativeLayout相对布局 a).第一类:属性值为true或false android:layout_centerHrizontal 水平居中 android:layout_center ...

  10. Yarn源码分析之如何确定作业运行方式Uber or Non-Uber?

    在MRAppMaster中,当MapReduce作业初始化时,它会通过作业状态机JobImpl中InitTransition的transition()方法,进行MapReduce作业初始化相关操作,而 ...