单例模式入门

单例模式

概述

  • 确保单一个类一个进程仅有一个实例,并提供一个全局访问点,适用于工具类、配置管理、线程池等场景

  • 避免重复创建对象浪费资源

  • 不需要实例化该类的对象,不能使用new关键词创建对象

  • 通常是自己创建自己的实例

与static的区别:

虽然static是实现单例的一种手段,

一个强调"共享性"

一个强调"唯一实例"

  • 单例模式是控制类的实例数量仅1个,通过实例访问

  • static是类级别的共享资源,无需实例即可访问

  • 单例模式在实例创建时开始,可手动销毁

  • static是随着类的加载而初始化,类的卸载而摧毁

  • 可通过私有构造、成员变量实现封装
优点:
  • 节省资源:对于数据库连接池,线程池等成本高的对象,避免重复创建对象,减少内存占用

  • 全局一致性,同一个对象,避免数据不一致

  • 通过getInstance()访问,可控,可添加控制逻辑(如线程安全检查)

  • 作为全局访问点,方便不同模块之间共享数据或资源,无需通过参数传递对象

应用场景:

  • 工具类:日志,配置,等

  • 资源管理器:池

  • 全局状态管理

    当一个类的实例必须全局唯一、多实例会导致问题或浪费,且需要全局访问

饿汉式

  • 类加载就会导致单实例对象创建

  • 创建在JVM的方法区(jdk8后称"元空间")

    静态变量、常量、类的结构信息(类名、父类、接口、方法表等)

    方法区的特点是:全局共享,所有线程共享

    public class Singleton {
    private static Singleton INSTANCE = new Singleton();
    //类内部用new创建一次实例,随着类加载
    private Singleton() {
    //私有构造,禁止new
    }
    public static void Singleton
    //全局访问点:返回内部创建的实例
    public static Singleton getINSTANCE() {
    return INSTANCE;
    }
    }
    //代码块方式
    public Singleton() {
    }
    private static Singleton INSTANCE;
    static {
    INSTANCE = new Singleton();
    }

懒汉式

  • 需要使用的时候才创建

  • 所以会出现线程不安全,创建多个对象的问题

    public Singleton() {
    }
    private static Singleton instance;

    public static Singleton getInstance() {
    if(instance ==null){
    instance = new Singleton();
    }//需要的时候调用getinstance创建实例
    return instance;
    }

    针对线程不安全所以要加个锁
    public static synchronized Singleton getInstance(){
    if(instance==null){
    instance=new Singleton();
    }
    }

java 复制代码
//将锁加在方法里面,也会创建多个实例  
public Singleton() {
    }
​
    private static Singleton instance;
​
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                instance = new Singleton();
            }
        }
        return instance;
    }
java 复制代码
//双重检查加锁
package test.Single;
public class Singleton {
    public Singleton() {
    }
​
    private static Singleton instance;
​
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                if(instance==null){
                instance = new Singleton();}
​
            }
        }
        return instance;
    }
}

静态内部类创建实例

借助了静态内部类不会随着JVM加载类而加载,而是和外部类相互独立,使用内部类的时候才会被加载

此时JVM保证内部类的唯一性

java 复制代码
public class Singleton {
    private Singleton() {
    }
        private static class staticSingleton{
            private static Singleton instance = new Singleton();
​
        }
    public static Singleton getInstance(){
        return staticSingleton.instance;
    }
}
java 复制代码
使用枚举类创建静态内部类

public enum Singleton1 {
    INSTANCE;
    public static Singleton1 getInstance(){
        return INSTANCE;
    }
}
​
相关推荐
重生之我是Java开发战士5 天前
【Java SE】多线程(三):单例模式,阻塞队列,线程池与定时器
java·javascript·单例模式
许彰午6 天前
34_Java设计模式之单例模式
java·单例模式·设计模式
罗超驿8 天前
10.Java单例模式全解析:饿汉式与懒汉式实现及线程安全深度剖析
安全·单例模式·javaee
布朗克1688 天前
33 设计模式精讲
java·单例模式·设计模式
雨浓YN9 天前
基于设计模式的Winform软件框架-01Xml\Log\Ini日志(单例模式+生产者消费者模式)
单例模式·设计模式
仙俊红9 天前
Java 单例模式:类里面为什么可以有自己类型的字段?
java·开发语言·单例模式
swordbob10 天前
prototype 注入到 singleton 里,prototype是否还是线程安全的
安全·spring·单例模式·原型模式
谁似人间西林客11 天前
工业大数据实战:看中国智造如何用数据驱动效率革命
大数据·单例模式
张小姐的猫11 天前
【Linux】多线程 —— 线程池 | 单例模式 | 常见锁
linux·运维·服务器·c++·单例模式·设计模式·策略模式
Java面试题总结12 天前
双重检验锁的单例模式在高并发下的可见性问题
单例模式