Java 实现一个单例模式_Java实现单例模式的两种方式

Java 实现一个单例模式_Java实现单例模式的两种方式单例模式在实际开发中有很多的用途,比如我们在项目中常用的工具类,数据库等资源的连接类。这样做的好处是避免创建多个对象,占用内存资源,自始自终在内存中只有一个对象为我们服务。单例对象一般有两种实现方式。懒汉式和饿汉式。饿汉式实现如下:packagecom.day05;/***饿汉式**@authorAdministrator**/publicclassSingle{//定义一个个私有静态本类对…

大家好,又见面了,我是你们的朋友全栈君。

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

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

饿汉式实现如下:

packagecom.day05;/*** 饿汉式

*

*@authorAdministrator

**/

public classSingle {//定义一个个私有静态本类对象变量,饿汉式是一上来就给该变量进行初始化,加上final是让s这个对象引用始终保持不变,更为严谨

private static final Single s = newSingle();//构造方法私有化,让外部无法通过new 关键字来进行创建对象

privateSingle() {

}//暴露出一个外界可以获取该类对象的公共静态方法

public staticSingle getInstance() {returns;

}

}

测试类:

packagecom.day05;public classSingleDemo {public static voidmain(String[] args) {

Single s1=Single.getInstance();

Single s2=Single.getInstance();

System.out.println(s1==s2);

}

}

运行结果:

true

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

懒汉式实现如下:

packagecom.day05;/*** 懒汉式

*

*@authorAdministrator

**/

public classSingle {//定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化

private static Single s = null;//构造方法私有化,让外部无法通过new 关键字来进行创建对象

privateSingle() {

}//暴露出一个外界可以获取该类对象的公共静态方法

public staticSingle getInstance() {if (s == null)

s= newSingle();returns;

}

}

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

packagecom.day05;/*** 懒汉式

*

*@authorAdministrator

**/

public classSingle {//定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化

private static Single s = null;//构造方法私有化,让外部无法通过new 关键字来进行创建对象

privateSingle() {

}//暴露出一个外界可以获取该类对象的公共静态方法

public staticSingle getInstance() {if (s == null) {try{

Thread.sleep(1000L);

}catch(InterruptedException e) {

e.printStackTrace();

}

s= newSingle();

}returns;

}

}

测试类如下:

packagecom.day05;public class SingleDemo implementsRunnable {public static voidmain(String[] args) {

SingleDemo sd= newSingleDemo();newThread(sd).start();newThread(sd).start();

}

@Overridepublic voidrun() {

Single s=Single.getInstance();

System.out.println(s);

}

}

运行结果如下:

com.day05.Single@4081b5a4

com.day05.Single@64dcdaac

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

问题分析:

67d0283fdb9526eb0f12a5d0788997a0.png

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

packagecom.day05;/*** 懒汉式

*

*@authorAdministrator

**/

public classSingle {//定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化

private static Single s = null;//构造方法私有化,让外部无法通过new 关键字来进行创建对象

privateSingle() {

}//暴露出一个外界可以获取该类对象的公共静态方法

public static synchronizedSingle getInstance() {if (s == null) {try{

Thread.sleep(1000L);

}catch(InterruptedException e) {

e.printStackTrace();

}

s= newSingle();

}returns;

}

}

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

packagecom.day05;/*** 懒汉式

*

*@authorAdministrator

**/

public classSingle {//定义一个个私有静态本类对象变量,懒汉式是先赋值为null,当需要的时候在初始化

private static Single s = null;//构造方法私有化,让外部无法通过new 关键字来进行创建对象

privateSingle() {

}//暴露出一个外界可以获取该类对象的公共静态方法//这里需要注意的使静态共享数据使用的使该类的字节码对象即Single.class

public staticSingle getInstance() {//这里增加了一次判断,可以少一次进行锁的处理

if (s == null) {synchronized (Single.class) {if (s == null) {try{

Thread.sleep(1000L);

}catch(InterruptedException e) {

e.printStackTrace();

}

s= newSingle();

}

}

}returns;

}

}

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/156950.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • 用我的亲身经历来告诉你如何自学Java?

    用我的亲身经历来告诉你如何自学Java?在我的另一篇博文——【一年总结】记我的大一生活中,我其实已经谈过了我的大一经历,但是对于我自学Java的地方往往都是一笔带过的,因为那篇博文主要是想记录我那时所思所想,表达的主要是一种情感,而非“攻略”。所以我觉得我有必要另起一篇来专门讲讲我是如何一步一步自学Java的。

  • cbow和skipgram适用于什么场景?_gram矩阵

    cbow和skipgram适用于什么场景?_gram矩阵在cbow方法中,是用周围词预测中心词,从而利用中心词的预测结果情况,使用GradientDesent方法,不断的去调整周围词的向量。当训练完成之后,每个词都会作为中心词,把周围词的词向量进行了调整,这样也就获得了整个文本里面所有词的词向量。要注意的是,cbow的对周围词的调整是统一的:求出的gradient的值会同样的作用到每个周围词的词向量当中去。可以看到,cbow预测行为的次数跟整个文本的…

  • 360天擎卸载方法[通俗易懂]

    360天擎卸载方法[通俗易懂]具体方法如下:1、先找到360天擎软件的安装文件夹,通常是:C:\ProgramFiles(x86)\360\360Safe\EntClient\conf2、先下载我们使有记事本打开目录下的EntBase.dat文件[base]persistent_connetion=closeshow_tip=1net_env=1communication_interval=900[api_frequency]checkupdate=180get_client_tasks=180getconf=

  • 帝国cms“建立目录不成功,请检查目录权限”的解决方法

    帝国cms“建立目录不成功,请检查目录权限”的解决方法

    2021年11月17日
  • 零基础入门STM32编程(二)

    零基础入门STM32编程(二)前情回顾上篇文章初步认识了STM32,了解了STM32的分类及型号命名规则,本篇内容继续STM32串口编程入门学习。一走进STM321.1STM32系列STM32单片机分为四类五种,四类介绍参见上一篇文章,本文仅阐述五个系列,分别为:STM32F0系列是针对8位和16位微控制器的32位MCU替代产品,对于工程人员来说非常有吸引力,他们希望其产品平台不会过时,并可围绕工业标准ArmCortex-M0内核进行标准化开发。 STM32G0系列是基于ArmCortex-M0+核心的新..

  • assert函数解析[通俗易懂]

    assert函数解析[通俗易懂]一、assert是宏明确一点:在C中,ASSERT是宏而不是函数。assert()是一个调试程序时经常使用的宏。在程序运行时它计算括号内的表达式。如果表达式为FALSE(0),程序将报告错误,并终止执行。如果表达式不为0,则继续执行后面的语句。这个宏通常用来判断程序中是否出现了明显非法的数据,如果出现就终止程序以免导致严重后果,同时反馈错误发生“地点”。

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号