进阶之路:Java设计模式---单例模式

el/2024/4/19 22:41:16
package com.example.administrator.javadesignmode.single_mode;public class SingleInstance {private static SingleInstance singleInstance;private SingleInstance(){}/*** 饿汉式* 多个线程调用时,会产生多个对象,线程不安全* @return*/public static SingleInstance getInstanceA(){if (singleInstance == null){singleInstance = new SingleInstance();}return singleInstance;}/*** 加入synchronized* 优点:解决多线程问题,避免多线程调用产生多个对象* 缺点:不高效,任何时候只能有一个线程调用getInstanceB()函数,每次调用都会进行同步,但是同步操作只是在第一次调用时才需要* @return*/public static synchronized SingleInstance getInstanceB(){if (singleInstance == null){singleInstance = new SingleInstance();}return singleInstance;}/*** 双重校验锁* 在同步块内、外各有一次校验instance == null* 优点:* 同步块外校验:避免已经有实例对象后每次调用时再次同步* 同步块内校验:避免多个线程同时通过外部判断进入后产生多个对象* 缺点:* singleInstance = new SingleInstance()为非原子操作,可能会崩溃,JVM在执行该句代码时做了三步操作:* 1.给singleInstance分配内存* 2.调用SingleInstance构造函数来初始化成员变量* 3.将singleInstance对象指向分配的内存空间* 在JVM的即时编译中存在指令重排序的优化,2.3步不能保证顺序执行,最终的执行顺序可能是1-2-3,也可能是1-3-2* 如果是1-3-2,在3执行完毕,2没有执行时,被线程二抢占,这时singleInstance为非空(但是却没有初始化),线程二直接返回singleInstance,然后使用,然后报错* 解决:* 将 instance 变量声明成 volatile ,禁止指令重排序优化,也就是说取得操作只能等1-2-3或者1-3-2完成之后才会执行* @return*/public static SingleInstance getInstanceC(){if (singleInstance == null){synchronized (SingleInstance.class){if (singleInstance == null){singleInstance = new SingleInstance();}}}return singleInstance;}/*** 饿汉式:优先创建实例对象,不存在同步问题和指令重排序问题* 缺点:* 在类加载时就被初始化,不符合懒加载* 如果实例的创建需要依赖初始参数,则同样不适用* private static SingleInstance singleInstance = new SingleInstance();* @return*/public static SingleInstance getInstanceD(){return singleInstance;}/*** 静态内部类形式* 使用java本身机制保证线程安全* 静态单例对象没有作为SingleInstance的成员变量直接实例化,因此在类加载时* 不会实例化instance对象,第一次调用getInstanceE()时将加载内部类SingleInstanceHolder* 此时会初始化内部类里面定义的INSTANCE对象,由java虚拟机保证线程安全,确保成员变量只实例化一次* 同时不具有同步锁,性能不会受到影响*/private static class SingleInstanceHolder{private static final SingleInstance INSTANCE = new SingleInstance();}public static SingleInstance getInstanceE(){return SingleInstanceHolder.INSTANCE;}}

代码地址


http://www.ngui.cc/el/5181700.html

相关文章

进阶之路:Java设计模式---Builder模式

什么是Builder模式? 将一个复杂对象的构建与它的表示相分离,使得同样的构建过程可以创建不同的对象。一步一步构建一个复杂的对象,它允许用户只通过指定复杂对象的内容和类型就可以构建他们,用户不需要知道内部的具体构建细节。 …

一天或者几天一个小算法---Fibonacci(菲波那切)数列

题目 输入n,求出Fibonacci数列第n项的数值 思路 1.Fibonacci数列是什么 F(0) 0; F(1) 1; F(n) F(n-1) F(n-2)2.实现开始 Fibonacci数列的第0、1、2项为基础数值,后面所有的数值都由这三个得来,前三个数的值可以判断得出,后…

进阶之路:Java设计模式---适配器模式

适配器模式定义: 适配器模式就是把一个类的接口变为另一个类期望的接口,从而让原来两个因接口不匹配而无法一起工作的两个类能在一起工作 适配器模式: 类适配器模式和对象适配器模式 类适配器模式实现: /*** 目标类*/ public…

一天或者几天一个小算法---数列排序方式

/*** 数字排序*/ public class Algorithm5 {/*** 冒泡排序* 自认为是最简单的一个排序方法,只是拿出一个数,大的放后,小的放前就可以了* 时间复杂度为O(n2)*/public static int [] sortOfBubble(int [] valueArray){int temp;for (int i 0 ;…

进阶之路:Java设计模式---外观模式

外观模式定义: 外部与子系统间进行通信时,需要通过一个统一的外观进行,为子系统的一系列接口提供一个统一的界面。外观模式定义了一个高层接口,这个接口使子系统更易使用。外观模式又称为门面模式,它是一种对象型结构…

Android崩溃日志收集---CrashHandler,给你自残的理由

使用CrashHandler的目的 测试人员过来告诉你,“你写的代码崩溃了!” “what ? ! 我写的代码怎么可能会崩溃?!你复现一下,污蔑我的话信不信自残给你看” “哎,这次怎么好了,什么情况?” 虽然说这种几率性bug可以晚点解决,但是终归是要解决的,但是复现又有点困难,那…

关于线程安全最通俗易懂的解释

下面的博客是在博客园上面看到的大佬写的,写的很好,通俗易懂,作为一个萌新看完以后促使我对线程安全有了新的理解。 博客未经博主的同意,所以未擅自转载,下面附快速通道 点次进入博客内容

Android 获取目录下所有文件信息(名字、大小)

private void getFiles(){File file = new File(GlobalConstant.POSLOGPATH);File[] files = file.listFiles();for (int i = 0; i < files.length; i++) {File childFile = files[i]

Java开发人员最常犯的9个错误

1.Array转ArrayList 当需要把Array转成ArrayList的时候&#xff0c;开发人员经常这样做&#xff1a; String [] strArray {"0" , "1" , "2"}; List<String> list Arrays.asList(intArray);Arrays.asList会返回一个ArrayList&#xff0…

Fragment的show()、hide()和attach()、detach()切换方式生命周期过程

1.使用show()、hide()方式进行fragment切换 @Overridepublic void onClick(View view) {switch (view.getId()){case R.id.main_home:switchFragmentWithShowHide(0);break;case R.id.main_shop:switchFragmentWithShowHide(1);break;case R.id.main_community:switchFragmentW…