Java实现单例模式之饿汉式、懒汉式、枚举式
作者:我只是一名程序员 发布时间:2022-12-17 17:50:13
标签:java,单例模式
单例模式的实现(5种)
常用:
饿汉式(线程安全,调用效率高,但是不能延时加载)
懒汉式(线程安全,调用效率不高,可以延时加载)
其他:
双重检测锁式(由于jvm底层内部模型原因,偶尔会出问题,不建立使用)
静态内部类式(线程安全,调用效率高,但是可以延时加载)
枚举单例(线程安全,调用效率高,不能延时加载)
饿汉式单例具体代码如下:
package com.lcx.mode;
/**
*
* 饿汉式单例,不管以后用不用这个对象,我们一开始就创建这个对象的实例,
* 需要的时候就返回已创建好的实例对象,所以比较饥饿,故此叫饿汉式单例。
* @author qq1013985957
*
*/
public class SingletonHanger {
private static final SingletonHanger instance = new SingletonHanger();
private SingletonHanger() {
}
public static SingletonHanger getInstance(){
return instance;
}
}
/**
* 懒汉汉式单例,在需要单例对象的时候,才创建唯一的单例对象,以后再次调用,返回的也是第一创建的单例对象
* 将静态成员初始化为null,在获取单例的时候才创建,故此叫懒汉式。
* @author qq1013985957
*
*/
class SingletonLazy{
private static SingletonLazy instance = null;
private SingletonLazy() {
}
/**
* 此方法实现的单例,无法在多线程中使用,多线可以同时进入if方法,会导致生成多个单例对象。
* @return
*/
public static SingletonLazy getInstance1(){
if(instance==null){
instance = new SingletonLazy();
}
return instance;
}
/**
* 大家都会想到同步,可以同步方法实现多线程的单例
* 但是这种方法不可取,严重影响性能,因为每次去取单例都要检查方法,所以只能用同步代码块的方式实现同步。
* @return
*/
public static synchronized SingletonLazy getInstance2(){
if(instance==null){
instance = new SingletonLazy();
}
return instance;
}
/**
* 用同步代码块的方式,在判断单例是否存在的if方法里使用同步代码块,在同步代码块中再次检查是否单例已经生成,
* 这也就是网上说的 双重检查加锁的方法
* @return
*/
public static synchronized SingletonLazy getInstance3(){
if(instance==null){
synchronized (SingletonLazy.class) {
if(instance==null){
instance = new SingletonLazy();
}
}
}
return instance;
}
}
/**
*
* 使用枚举实现单例模式,也是Effective Java中推荐使用的方式
* 根据具体情况进行实例化,对枚举不熟悉的同学,可以参考我的博客 JAVA 枚举类的初步理解。
* 它的好处:更加简洁,无偿提供了序列化机制,绝对防止多次实例化,即使面对复杂的序列和反射攻击。
* @author qq1013985957
*
*/
enum SingletionEnum{
SingletionEnum("单例的枚举方式");
private String str ;
private SingletionEnum(String str){
this.setStr(str);
}
public String getStr() {
return str;
}
public void setStr(String str) {
this.str = str;
}
}
以上的单例模式就不测试,大家可以去测试,判断对象的hashcode是否一致来判断是否为同一个对象。
恶汉式、懒汉式的方式还不能防止反射来实现多个实例,通过反射的方式,设置ACcessible.setAccessible方法可以调用私有的构造器,可以修改构造器,让它在被要求创建第二个实例的时候抛出异常。
其实这样还不能保证单例,当序列化后,反序列化是还可以创建一个新的实例,在单例类中添加readResolve()方法进行防止。
懒汉汉式单例代码如下:
package com.lcx.mode;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
/**
* 懒汉汉式单例,在需要单例对象的时候,才创建唯一的单例对象,以后再次调用,返回的也是第一创建的单例对象
* 将静态成员初始化为null,在获取单例的时候才创建,故此叫懒汉式。
* @author qq1013985957
*
*/
public class Singleton implements Serializable{
/**
*
*/
private static final long serialVersionUID = -5271537207137321645L;
private static Singleton instance = null;
private static int i = 1;
private Singleton() {
/**
* 防止反射攻击,只运行调用一次构造器,第二次抛异常
*/
if(i==1){
i++;
}else{
throw new RuntimeException("只能调用一次构造函数");
}
System.out.println("调用Singleton的私有构造器");
}
/**
* 用同步代码块的方式,在判断单例是否存在的if方法里使用同步代码块,在同步代码块中再次检查是否单例已经生成,
* 这也就是网上说的 双重检查加锁的方法
* @return
*/
public static synchronized Singleton getInstance(){
if(instance==null){
synchronized (Singleton.class) {
if(instance==null){
instance = new Singleton();
}
}
}
return instance;
}
/**
*
* 防止反序列生成新的单例对象,这是effective Java 一书中说的用此方法可以防止,具体细节我也不明白
* @return
*/
private Object readResolve(){
return instance;
}
public static void main(String[] args) throws Exception {
test1();
test2();
}
/**
* 测试 反序列 仍然为单例模式
* @throws Exception
*/
public static void test2() throws Exception{
Singleton s = Singleton.getInstance();
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(new File("E:\\Singleton.txt")));
objectOutputStream.writeObject(s);
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(new File("E:\\Singleton.txt")));
Object readObject = objectInputStream.readObject();
Singleton s1 = (Singleton)readObject;
System.out.println("s.hashCode():"+s.hashCode()+",s1.hashCode():"+s1.hashCode());
objectOutputStream.flush();
objectOutputStream.close();
objectInputStream.close();
}
/**
* 测试反射攻击
* @throws Exception
*/
public static void test1(){
Singleton s = Singleton.getInstance();
Class c = Singleton.class;
Constructor privateConstructor;
try {
privateConstructor = c.getDeclaredConstructor();
privateConstructor.setAccessible(true);
privateConstructor.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}
验证反射攻击结果:
如果不添加readResolve方法的结果:
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


猜你喜欢
- 一、事件背景个人感觉自己做性能测试,可以说是轻车熟路了,而且工作多年一直都是这一套测试思路及体系,从未质疑过自己,也许是狮子座的迷之自信吧!
- 最可怕的不是犯错而是一直都没发现错误,直到现在我才知道自己对类变量的理解有问题。大概可能也许是因为不常用类变量的原因吧,一直没有发现这个问题
- 连接操作redisSpring Boot中操作redis还是需要使用相关的启动器<dependency><groupId&
- Android中有两种主要方式使用Service,通过调用Context的startService方法或调用Context的bindServ
- try { // 方
- flutter material widget组件之信息展示组件,供大家参考,具体内容如下widget分为两类:widgets librar
- 参考 java查找无向连通图中两点间所有路径的算法,对代码进行了部分修改,并编写了测试用例。算法要求:1. 在一个无向连通图中求出
- 接收 / 返回文本消息①接收/返回文本消息原理说明当普通微信用户向公众账号发消息时,微信服务器将POST消息的XML数据包到开发者填写的UR
- jasperreport导出的pdf每页显示的记录太少主要是确保Details的高度与Details中Field Text的高度一致。jas
- 本文实例讲述了Android下2d物理引擎Box2d用法。分享给大家供大家参考。具体如下:程序运行的时候需要加载Jbox2d的库,可到以下地
- 在上一节,我们已经完成了项目的整体技术架构设计和具体的数据库设计,接下来,我们搭建整体的开发框架。开发工具选用Idea。开发工具只是为了提高
- 这篇文章主要介绍了java io读取文件操作代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友
- WCF实例(带步骤) <xmlnamespace prefix ="o" ns ="urn:schema
- Spring Cloud Gateway是Spring 官方基于Spring 5.0、Spring Boot 2.0和Project Rea
- 目录源码实现基本流程ThreadLoalMap数据结构Hash冲突及解决ThreadLocal内存泄露内存引用链路引用类型为什么使用弱引用而
- 一、题目描述题目实现:在进行网络编程时,由于进行网络连接是比较消耗资源的,因此,可以对连接的等待时间进行设置,如果在规定的时间没有进行连接,
- 工厂方法模式定义: Define an interface for creating an object, but let subclass
- 用户提出一个需求:当修改产品价格的时候,需要记录操作日志,什么时候做了什么事情。想必这个案例,只要是做过应用系统的小伙伴们,都应该遇到过吧?
- Spring Security提供如下几种认证机制Username & PasswordOAuth2.0 LoginSAML 2.0
- 网上有很多的OkHttp的教程,但是并没有一个是关于如何OkHttp处理重定向的。这里的处理重定向的意思是:把重定向请求拦截下来,然后我们自