热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:java设计模式单例模式(干货)

篇首语:本文由编程笔记#小编为大家整理,主要介绍了java设计模式-单例模式(干货)相关的知识,希望对你有一定的参考价值。深度讲解23种设计模式,力争每种设计模式

篇首语:本文由编程笔记#小编为大家整理,主要介绍了java设计模式 - 单例模式(干货)相关的知识,希望对你有一定的参考价值。


  深度讲解23种设计模式,力争每种设计模式都刨析到底。废话不多说,开始第一种设计模式 - 单例。

  作者已知的单例模式有8种写法,而每一种写法,都有自身的优缺点。

1,使用频率最高的写法,废话不多说,直接上代码

/**
* @author xujp
* 饿汉式 静态变量 单例
*/

public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private final static Singleton instance = new Singleton();

private Singleton(){}
public static Singleton getSingleton(){
return instance;
}

private String tmp;

public String getTmp() {
return tmp;
}

public void setTmp(String tmp) {
this.tmp = tmp;
}
}

 new Singleton() 的执行时机 - > 类加载时

 这种方法是最通用的单例实现,也是笔者常用的,但这种方法有一些缺点

 1)内存方面,如果单例中的内容很多,会在类加载时,就占用java虚拟机(这里专指HotSpot)空间。

 2)序列化以及反序列化问题,如果这个单例类实现了序列化接口Serializable,那么可以通过反序列化来破坏单例。

 通过反序列化破坏单例:

public static void main(String[] args) throws IOException, ClassNotFoundException {
Singleton singleton=null;
Singleton singletOnNew=null;

singleton=Singleton.getSingleton();

singleton.setTmp("123");

ByteArrayOutputStream bos=new ByteArrayOutputStream();
ObjectOutputStream oos=new ObjectOutputStream(bos);
oos.writeObject(singleton);

ByteArrayInputStream bis=new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois=new ObjectInputStream(bis);
singletOnNew= (Singleton) ois.readObject();

singleton.setTmp("456");

System.out.println(singletonNew.getTmp());
System.out.println(singleton.getTmp());
System.out.println(singleton==singletonNew);
}

  输出结果为:

  false

  123

  456

  从这里例子中我们可以看到单例被破坏了,也就不能保证单例的唯一性。

2,第一种方案的变种

/**
* @author xujp
* 饿汉式 静态代码块 单例
*/
public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private final static Singleton instance;

static {
instance = new Singleton();
}

private Singleton(){}

public static Singleton getSingleton(){
return instance;
}
}

 其实这种方法和第一种方法,几乎没有什么区别。

3,线程不安全的写法 - 1

/**
* @author xujp
* 懒汉式 单例 线程不安全
*/
public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private static Singleton instance;

private Singleton(){}

public static Singleton getSingleton(){
if(null != instance) {
instance = new Singleton();
}
return instance;
}
}

 这种写法,虽然实现了懒加载,节省了内存,但线程不安全。

 假设有两个线程,并假设 new Singleton() 耗时2秒,0秒时,线程1执行new,然后去等待,1秒时,线程2执行if判断,

这个时候判断结果就是true,这样就会出现两个Singleton对象,完美破坏掉了单例。

4,线程不安全的写法 - 2

/**
* @author xujp
* 懒汉式 单例 代码块加锁 线程不安全
*/
public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private static Singleton instance;

private Singleton(){}

public static Singleton getSingleton(){
if(null != instance) {
synchronized (Singleton.class) {
instance = new Singleton();
}
}
return instance;
}
}

 这种写法虽然在new Single()时,增加了锁,但这个锁,并不能阻止单例被破坏,所以这种写法错误。

 同样,假设有两个线程,线程1执行到synchronized时,线程2执行if判断,这个时候判断结果就是true,

这样就会出现两个Singleton对象,同样完美破坏掉了单例。

5,线程安全,但资源消耗过多

/**
* @author xujp
* 懒汉式 单例 方法加锁 线程不安全
*/
public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private static Singleton instance;

private Singleton(){}

synchronized public static Singleton getSingleton(){
if(null != instance) {
instance = new Singleton();
}
return instance;
}
}

 这种写法确实能够保证线程安全,但synchronized属于方法锁,而方法锁回锁定对象,导致性能低下。

6,相对完美的写法 - 1

/**
* @author xujp
* 懒汉式 单例 代码加锁 线程安全
*/
public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private static Singleton instance;

private Singleton(){}

public static Singleton getSingleton(){
if(null != instance) {
synchronized (Singleton.class) {
if(null != instance) {
instance = new Singleton();
}
}
}
return instance;
}
}

双检查这种写法,在多线程问题上,属实没有问题,synchronized也没有锁定对象,而且也优化了锁资源开销问题。

7,相对完美的写法 - 2

/**
* @author xujp
* 懒汉式 单例 静态内部类 线程安全
*/
public class Singleton implements Serializable {

private static final long serialVersiOnUID= 1L;

private static class SingletonInstance{
private static Singleton instance = new Singleton();
}
private Singleton(){}

public static Singleton getSingleton(){
return SingletonInstance.instance;
}
}

使用静态内部类来实现单例,主要借助JVM机制,静态内部类初始化的时候,其他线程无法进入,从而避免了多线程问题。

而且静态内部类不会直接初始化,从而减轻了内存开销。

8,完美写法

/**
* @author xujp
* 枚举实现单例
*/
public enum Singleton {
SINGLETON;
private String property = "hello ca fe ba be";
public void doSomeThing(){
System.out.println(property);
}
}

这种写法用枚举解决多线程问题,而且时唯一一种解决序列化问题的写法。

改写法出自大神Josh Bloch,如果有兴趣可以去查看一下他的资料。

总结:

1,1和2写法虽然是饿汉式,没有实现懒加载,也没有100%保证单例,但却是我们最常用的写法,

 因为,单例对象通常占用空间不会很大,而且程序都由程序员自己管理,被反序列的危险性不高。

2,3和4写法实现了懒加载,减少了内存开销,但不能使用,因为多线程开发,是我们常见的开发。

3,5写法使用了方法锁,会将对象锁住,会导致性能大打折扣。

4,6和7写法,懒加载、性能都非常完美,缺点只有一个,那就是序列化问题。

5,8写法,笔者暂未发现缺点。

实际开发中,无论是使用1、2写法,还是使用6、7写法,亦或是使用8写法,都是可以的。

 

  

 


推荐阅读
  • 本文介绍了Swing组件的用法,重点讲解了图标接口的定义和创建方法。图标接口用来将图标与各种组件相关联,可以是简单的绘画或使用磁盘上的GIF格式图像。文章详细介绍了图标接口的属性和绘制方法,并给出了一个菱形图标的实现示例。该示例可以配置图标的尺寸、颜色和填充状态。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 纠正网上的错误:自定义一个类叫java.lang.System/String的方法
    本文纠正了网上关于自定义一个类叫java.lang.System/String的错误答案,并详细解释了为什么这种方法是错误的。作者指出,虽然双亲委托机制确实可以阻止自定义的System类被加载,但通过自定义一个特殊的类加载器,可以绕过双亲委托机制,达到自定义System类的目的。作者呼吁读者对网上的内容持怀疑态度,并带着问题来阅读文章。 ... [详细]
  • 基于Socket的多个客户端之间的聊天功能实现方法
    本文介绍了基于Socket的多个客户端之间实现聊天功能的方法,包括服务器端的实现和客户端的实现。服务器端通过每个用户的输出流向特定用户发送消息,而客户端通过输入流接收消息。同时,还介绍了相关的实体类和Socket的基本概念。 ... [详细]
  • 本文介绍了九度OnlineJudge中的1002题目“Grading”的解决方法。该题目要求设计一个公平的评分过程,将每个考题分配给3个独立的专家,如果他们的评分不一致,则需要请一位裁判做出最终决定。文章详细描述了评分规则,并给出了解决该问题的程序。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • 本文介绍了一个题目的解法,通过二分答案来解决问题,但困难在于如何进行检查。文章提供了一种逃逸方式,通过移动最慢的宿管来锁门时跑到更居中的位置,从而使所有合格的寝室都居中。文章还提到可以分开判断两边的情况,并使用前缀和的方式来求出在任意时刻能够到达宿管即将锁门的寝室的人数。最后,文章提到可以改成O(n)的直接枚举来解决问题。 ... [详细]
  • Ihavethefollowingonhtml我在html上有以下内容<html><head><scriptsrc..3003_Tes ... [详细]
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • 本文讨论了在VMWARE5.1的虚拟服务器Windows Server 2008R2上安装oracle 10g客户端时出现的问题,并提供了解决方法。错误日志显示了异常访问违例,通过分析日志中的问题帧,找到了解决问题的线索。文章详细介绍了解决方法,帮助读者顺利安装oracle 10g客户端。 ... [详细]
  • 本文介绍了Android中的assets目录和raw目录的共同点和区别,包括获取资源的方法、目录结构的限制以及列出资源的能力。同时,还解释了raw目录中资源文件生成的ID,并说明了这些目录的使用方法。 ... [详细]
  • ***byte(字节)根据长度转成kb(千字节)和mb(兆字节)**parambytes*return*publicstaticStringbytes2kb(longbytes){ ... [详细]
  • 本文整理了Java面试中常见的问题及相关概念的解析,包括HashMap中为什么重写equals还要重写hashcode、map的分类和常见情况、final关键字的用法、Synchronized和lock的区别、volatile的介绍、Syncronized锁的作用、构造函数和构造函数重载的概念、方法覆盖和方法重载的区别、反射获取和设置对象私有字段的值的方法、通过反射创建对象的方式以及内部类的详解。 ... [详细]
  • 使用C++编写程序实现增加或删除桌面的右键列表项
    本文介绍了使用C++编写程序实现增加或删除桌面的右键列表项的方法。首先通过操作注册表来实现增加或删除右键列表项的目的,然后使用管理注册表的函数来编写程序。文章详细介绍了使用的五种函数:RegCreateKey、RegSetValueEx、RegOpenKeyEx、RegDeleteKey和RegCloseKey,并给出了增加一项的函数写法。通过本文的方法,可以方便地自定义桌面的右键列表项。 ... [详细]
author-avatar
beng83790si
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有