java23种设计模式(1)-单例模式

简介: java23种设计模式(1)-单例模式
简要说明

1、单例模式保证了 系统内存中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使

用单例模式可以提高系统性能

2、当想实例化一个单例类的时候,必须要记住使用相应的获取对象的方法,而不是使用 new

3、单例模式 使用的场景:需要 频繁的进行创建和销毁的对象、创建对象时耗时过多或耗费资源过多(即:重量级对象),但又经常用到的对象、 工具类对象、频繁访问数据库或文件的对象(比如 数据源、session 工厂等)

1、饿汉式单例模式

package singleton;
/**
 * 饿汉式单例模式
 */
public class Singleton {
    //方法区静态常量池初始化对象
    private static final Singleton singleton = new Singleton();
    //构造方法私有化,不让外部new
    private Singleton(){}
    public static Singleton getInstance(){
        return singleton;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String Args[]){
        Singleton p1 = Singleton.getInstance();
        Singleton p2 = Singleton.getInstance();
        System.out.println(p1 == p2);
    }
}

饿汉式在类进行初始化的时候就将类实例创建好,不管是否使用。而在使用时直接调用就行,不需要进行判断,同时需要花费大量的空间,典型的空间换时间。

2、懒汉式单例模式

必须加锁,如果不加锁,使用懒汉式的多线程不安全

1,在对象之间加锁,缺点是每个对象都需要进行等待和判断,效率低

package singleton;
/**
 * 懒汉式:用的时候进行初始化
 */
public class Singleton2 {
    //不需要final,在加final时需要初始化
    private static Singleton2 singleton2;
    private Singleton2(){}
    public static synchronized Singleton2 getInstance(){
        //在此区间可能出现线程不安全
        //因此需要加锁使得线程同步,缺点是效率降低
        if(singleton2 == null){
            try{
                Thread.sleep(10);
            }catch(Exception e){
                e.printStackTrace();
            }
            singleton2 = new Singleton2();
        }
        return singleton2;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) throws Exception{
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                //从打印区间看看对象的hash码是否相同
                System.out.println(Singleton2.getInstance().hashCode());
            }).start();
        }
    }
}

2,改进,使用双重检查

package singleton;
public class Singleton3 {
    //不需要final,在加final时需要初始化
    private static Singleton3 singleton3;
    private Singleton3(){}
    public static  Singleton3 getInstance(){
        //在此区间可能出现线程不安全
        //因此需要加锁使得线程同步,缺点是效率降低
        if(singleton3 == null){
            //因此需要双重检查
            synchronized (Singleton3.class){
                if(singleton3 == null){
                    try{
                        Thread.sleep(10);
                    }catch(Exception e){
                        e.printStackTrace();
                    }
                    singleton3 = new Singleton3();
                }
            }
        }
        return singleton3;
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) throws Exception{
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                //从打印区间看看对象的hash码是否相同
                System.out.println(Singleton3.getInstance().hashCode());
            }).start();
        }
    }
}

3,使用比较完美的静态内部类实现单例模式

package singleton;
/**
 * 完美的写法
 */
public class Singleton4 {
    private Singleton4(){}
    //使用静态内部类
    private static class Singleton4Hold{
        private final static Singleton4 singleton4 = new Singleton4();
    }
    public static Singleton4 getInstance(){
        return Singleton4.getInstance();
    }
    public void m(){
        System.out.println("m");
    }
    public static void main(String[] args) throws Exception{
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                //从打印区间看看对象的hash码是否相同
                System.out.println(Singleton4.getInstance().hashCode());
            }).start();
        }
    }
}

4,枚举实现单例模式

简单高效,不仅可以解决线程同步的问题,还可以防止被序列化

防止被序列化的原因是因为枚举内部没有构造方法

package singleton;
public enum Singleton5 {
    INSTANCE;
    public void m(){}
    public static void main(String[] args) {
        for (int i = 0; i < 100; i++) {
            new Thread(()->{
                System.out.println(Singleton5.INSTANCE.hashCode());
            }).start();
        }
    }
}

5、总结

-单例对象 占用资源少,不需要延时加载,枚举 好于 饿汉

-单例对象 占用资源多,需要延时加载,静态内部类 好于 懒汉式

相关文章
|
17天前
|
设计模式 Java Spring
Java 设计模式之责任链模式:优雅处理请求的艺术
责任链模式通过构建处理者链,使请求沿链传递直至被处理,实现发送者与接收者的解耦。适用于审批流程、日志处理等多级处理场景,提升系统灵活性与可扩展性。
137 2
|
17天前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
175 0
|
3月前
|
设计模式 缓存 Java
Java设计模式(二):观察者模式与装饰器模式
本文深入讲解观察者模式与装饰器模式的核心概念及实现方式,涵盖从基础理论到实战应用的全面内容。观察者模式实现对象间松耦合通信,适用于事件通知机制;装饰器模式通过组合方式动态扩展对象功能,避免子类爆炸。文章通过Java示例展示两者在GUI、IO流、Web中间件等场景的应用,并提供常见陷阱与面试高频问题解析,助你写出灵活、可维护的代码。
|
16天前
|
设计模式 算法 搜索推荐
Java 设计模式之策略模式:灵活切换算法的艺术
策略模式通过封装不同算法并实现灵活切换,将算法与使用解耦。以支付为例,微信、支付宝等支付方式作为独立策略,购物车根据选择调用对应支付逻辑,提升代码可维护性与扩展性,避免冗长条件判断,符合开闭原则。
184 35
|
16天前
|
设计模式 消息中间件 传感器
Java 设计模式之观察者模式:构建松耦合的事件响应系统
观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。
160 8
|
6月前
|
设计模式 缓存 安全
【设计模式】【创建型模式】单例模式(Singleton)
一、入门 什么是单例模式? 单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。它常用于需要全局唯一对象的场景,如配置管理、连接池等。 为什么要单例模式? 节省资源 场景:某些对象创
204 15
|
6月前
|
设计模式 缓存 安全
【高薪程序员必看】万字长文拆解Java并发编程!(8):设计模式-享元模式设计指南
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的经典对象复用设计模式-享元模式,废话不多说让我们直接开始。
136 0
|
3月前
|
设计模式 安全 Java
Java设计模式(一):单例模式与工厂模式
本文详解单例模式与工厂模式的核心实现及应用,涵盖饿汉式、懒汉式、双重检查锁、工厂方法、抽象工厂等设计模式,并结合数据库连接池与支付系统实战案例,助你掌握设计模式精髓,提升代码专业性与可维护性。
|
3月前
|
设计模式 XML 安全
Java枚举(Enum)与设计模式应用
Java枚举不仅是类型安全的常量,还具备面向对象能力,可添加属性与方法,实现接口。通过枚举能优雅实现单例、策略、状态等设计模式,具备线程安全、序列化安全等特性,是编写高效、安全代码的利器。
|
8月前
|
设计模式 Java 数据安全/隐私保护
Java 设计模式:装饰者模式(Decorator Pattern)
装饰者模式属于结构型设计模式,允许通过动态包装对象的方式为对象添加新功能,提供比继承更灵活的扩展方式。该模式通过组合替代继承,遵循开闭原则(对扩展开放,对修改关闭)。