单例模式

本文介绍了Java中的单例模式实现,包括线程不安全的懒汉式、线程安全的懒汉式和饿汉式。懒汉式在多线程环境下可能导致多个实例,而线程安全的懒汉式通过同步解决了这一问题,但牺牲了性能。饿汉式在类加载时即创建实例,确保线程安全但可能提前占用资源。根据业务需求选择适合的单例实现方式,平衡启动速度和资源消耗。
摘要由CSDN通过智能技术生成

就算大雨让整座城市倾倒,上班照样会被算做迟到。

确保某一个类只有一个实例,而且自行实例化,并向整个系统提供这个实例
现在我们来举个栗子:
英雄联盟是很多人都玩过的一个游戏,盖伦(Garen)是英雄联盟中非常有代表性的英雄。英雄联盟的排位模式,规定了在一次游戏中的10个英雄只可以出现1个盖伦。此时,便可以选择用单例模式的方法来创建盖伦。

一、线程不安全的懒汉式单例

延迟加载的方式,只有使用的时候才会加载。在多个线程的情况下,可能会出现多个线程向Garen类请求静态方法getInstance(),导致生成两个garen对象

public class Garen{
	private static Garen garen;
	private Garen(){
	
	}
	
	public static Garen getInstance{
		if(garen==null){
			garen = new Garen();
		}
		return garen;
	}
	
	public void getBaron(){
		System.out.println("我是盖伦,我抢到了大龙");
	}
}

二、线程安全的懒汉式单例

  1. 为解决懒汉式单例模式的线程不安全问题,将静态方法getInstance()改造成同步方法,并向属性garen加入volatile关键字,能保证线程安全。
  2. 但缺点在于每次访问garen属性时数据都需要同步,会在一定程度上影响性能,而且会消耗更多的资源。
  3. 使用懒汉式,在启动的时候,会感觉到比饿汉式略快,因为并没有做对象的实例化。 但是在第一次调用的时候,会进行实例化操作,感觉上就略慢。
public class Garen{
	// 轻量级数据同步机制,保障garen对象在所有线程中同步
	private static volatile Garen garen;
	private Garen(){
	
	}
	
	public static synchronized Garen getInstance(){
		if(garen==null){
			garen = new Garen();
		}
		return garen;
	}

	public void getBaron(){
		System.out.println("我是盖伦,我抢到了大龙");
	}
}

三、线程安全的饿汉式单例

饿汉式单例,已经保障了线程安全,缺点在于还不需要创建garen对象时,对象已经被创建出来了。如果在构造方法里写了性能消耗较大,占时较久的代码,比如建立与数据库的连接,那么就会在启动的时候感觉稍微有些卡顿。

public class Garen{
	private static final Garen garen = new Garen();
	private Garen(){
	
	}
	
	public static Garen getInstance(){
		return garen;
	}
	
	public void getBaron(){
		System.out.println("我是盖伦,我抢到了大龙");
	}
}

小结:看业务需求,如果业务上允许有比较充分的启动和初始化时间,就使用饿汉式,否则就使用懒汉式

四、关于单例的延伸
如果我们玩的模式不是排位模式,而是匹配模式。那么就允许创建两个盖伦对象(蓝方一个紫方一个,但最多只有两个)。此时就需要用List将产生的单例存起来,并使用静态代码块进行初始化。

import java.util.ArrayList;
import java.util.List;
import java.util.Random;

public class Garen {
    private static List<Garen> list = new ArrayList<>(2);
    private static final int maxNum = 2;
    private String color;

    private Garen(String color) {
        this.color = color;
    }

    static {
        list.add(new Garen("blue"));
        list.add(new Garen("purple"));
    }

    public static Garen getInstance(){
        int index = new Random().nextInt(maxNum);
        return list.get(index);
    }

    public void getBaron(){
        System.out.println("我是"+color+"方的盖伦,我抢到了大龙");
    }
}

单例模式三要素

  1. 构造方法私有化
  2. 静态属性指向实例
  3. public static的getInstance()方法,返回第2步的静态属性

单例模式的使用场景

  1. 某类只要求生成一个对象
  2. 当对象需要被共享的场合,如Web中的配置对象,数据库连接池
  3. 当某类需要频繁实例化,而创建的对象又被频繁销毁,如多线程的线程池,网络连接池
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值