设计模式(持续更新中)
本文最后更新于 328 天前,其中的信息可能已经有所发展或是发生改变。

定义

确保一个类最多只有一个实例,并提供一个全局访问点

单例模式的四种写法:饿汉模式,懒汉模式,静态内部类,枚举

饿汉模式

即预处理模式,在类加载时直接创建并初始化单例对象,它依靠ClassLoader类加载,在程序启动时只加载一次,因此不存在线程安全问题。

public class Singleton {
    // 1.防止外部直接 new 对象破坏单例模式
    private Singleton() {}
    // 2.通过私有变量保存单例对象
    private static Singleton instance = new Singleton();
    // 3.提供公共获取单例对象的方法
    public static Singleton getInstance() {
        return instance;
    }
}

优点:实现简单,不存在线程安全问题。缺点:类加载时就创建对象,创建之后未被使用,就是造成了资源浪费的情况

public class Singleton {
    // 1.防止外部直接 new 对象破坏单例模式
    private Singleton() {}
    // 2.通过私有变量保存单例对象
    private static Singleton instance = new Singleton();
    // 3.提供公共获取单例对象的方法
    public static Singleton getInstance() {
        return instance;
    }
}

优点:实现简单,不存在线程安全问题。缺点:类加载时就创建对象,创建之后未被使用,就是造成了资源浪费的情况

懒汉模式

懒汉模式就是懒加载(延迟加载),指的是它只有在第一次被使用时,才会初始化

public class Singleton {
    // 1.防止外部直接 new 对象破坏单例模式
    private Singleton() {}
    // 2.通过私有变量保存单例对象
    private static volatile Singleton instance = null;
    // 3.提供公共获取单例对象的方法
    public static Singleton getInstance() {
        if (instance == null) { // 第一次效验
            synchronized (Singleton.class) {
                if (instance == null) { // 第二次效验
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}

懒汉模式使用的双重效验锁和 volatile 来保证线程安全的。

饿汉模式和懒汉模式实现步骤相同:

  1. 创建一个私有的构造方法,防止其他调用的地方直接 new 对象。
  2. 创建一个私有变量来保存单例对象
  3. 提供一个公共的方法返回单例对象

相对于饿汉模式,不会造成资源浪费,但写法复杂一些

静态内部类

静态内部类既能保证线程安全,又能保证懒加载,它只有在被调用的时候,才会通过 ClassLoader 机制来加载和初始化内部静态类,因此线程安全。

public class Singleton {
    // 1.防止外部直接 new 对象破坏单例模式
    private Singleton() {
    }

    // 2.静态内部类
    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    // 3.提供公共获取单例对象的方法
    public static final Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

枚举

枚举也是在第一次使用时,才会被 Java 虚拟机加载并初始化,所以它也是线程安全的,且是懒加载的。

public enum  EnumSingleton {
    INSTANCE;
    public EnumSingleton getInstance(){
        return INSTANCE;
    }
}

装饰者模式

什么是装饰者模式

装饰者模式是一种软件设计模式,它允许在不改变对象结构的情况下,动态地向对象添加额外的功能。该模式通过将对象包装在一个装饰器对象中,来实现对对象的功能扩展。这种模式通常用于需要在运行时动态地为对象添加功能的情况,而不是在编译时静态地定义对象的功能。

举个例子:比如一个毛坯房,可以使用灰色去装饰厨房,可以使用白色装饰客厅,但是不管使用多少颜料去装饰这个房间,房子的本质是不会发现变化的。

使用装饰者模式除了可以在不改变原类的前提下装饰对象之外,还可以随意组合各个装饰类,比如有三个装饰类,分别是给原类上红、黄、绿三种颜色。在使用时,可以将任意两种或任意三种装饰类搭配。接下来将会用代码演示上面的这个例子。

装饰者模式中的角色

装饰者模式中具备以下四种角色:

Component:一个抽象的组件类,在装饰者模式中扮演着最重要的角色。在给房子上色这个例子中,概念上的房子就是一个Component。如果还不理解,继续往下看。

ConcreteComponent:Component的具体实现,在本次的例子中就是商品房、公寓房等。

Decorator:一个通用的装饰接口或抽象类,内部保存了被装饰的Component对象。

ConcreteDecorator:具体的装饰类,在原类的基础上进行装饰。

装饰者模式案例

原类是房子,可装饰红色、黄色、绿色。接下来的代码就是对这个例子的体现:首先定义Component,也就是房子抽象类,有一个展示房子的方法:

public abstract class House {
    /**
     * 展示房子
     */
    public abstract void show();
}

接下来是ConcreteComponent,抽象房子的实现类,这里定义为商品房,ConcreteComponent可以有很多个

public class CommercialHouse extends House{
    @Override
    public void show() {
        System.out.println("这是一个商品房");
    }
}

Decorator是一个通用的抽象类,将房子作为入参传入,并实现同样的show方法

public class HouseDecorator extends House{
    private House house;

    public HouseDecorator(House house){
        this.house = house;
    }

    @Override
    public void show() {
        house.show();
    }
}
public class RedHouseDecorator extends HouseDecorator{

    public RedHouseDecorator(House house) {
        super(house);
    }
    @Override
    public void show() {
        super.show();
        System.out.println("装饰了红色");
    }
}

public class YellowHouseDecorator extends HouseDecorator{

    public YellowHouseDecorator(House house) {
        super(house);
    }
    @Override
    public void show() {
        super.show();
        System.out.println("装饰了黄色");
    }
}

public class GreenHouseDecorator extends HouseDecorator{

    public GreenHouseDecorator(House house) {
        super(house);
    }
    @Override
    public void show() {
        super.show();
        System.out.println("装饰了绿色");
    }
}


接下来就是使用了,前面已经提到了,装饰者模式除了可以在不改动原类的情况下增加功能,也可以随意组装各个装饰类:

public class Main {
    public static void main(String[] args) {
        House house = new CommercialHouse();
        //只装饰红色
        House redHouse = new RedHouseDecorator(house);
        redHouse.show();
        //装饰绿色和黄色
        House greenAndYellowHouse = new GreenHouseDecorator(new YellowHouseDecorator(house));
        greenAndYellowHouse.show();
        //三种颜色全部装饰
        House allHouse = new RedHouseDecorator(new GreenHouseDecorator(new YellowHouseDecorator(house)));
        allHouse.show();
    }
}

在上面的测试代码中,既可以只有一种颜色的装饰,也可以有两种颜色的装饰,或者是三种颜色的装饰,而原本的商品房这个类没有经过任何修改。如果用继承去实现,每种组装方式都需要新建一个类,装饰者模式的优势就体现出来了。

装饰者模式在源码中的应用

看了很多装饰者模式在源码中的应用,感觉Java.IO中的流在装饰者模式的使用上最经典。以InputStream为例子,InputStream有很多的实现类:

FileInputStream:实现文件的读取。

DataInputStream:读取各种基本数据类型的数据。

BufferedInputStream:可缓存的文件流。

ObjectInputStream:读取对象的文件流。

其他的实现还有很多很多,这里的实现就使用了装饰者模式,保证InputStream不变的前提下,增加其他功能。想象一下,如果要同时实现文件读取和可缓存,那么就可以这样写:

new BufferedInputStream(new FileInputStream(""));

是不是和第三节的例子一模一样,在代码的实现上,其实也大同小异,InputStream是一个抽象类,定义了read方法,代码作了精简:

new DataInputStream(new FileInputStream(""));

是不是和第三节的例子一模一样,在代码的实现上,其实也大同小异,InputStream是一个抽象类,定义了read方法,代码作了精简

public abstract class InputStream implements Closeable{
    public abstract int read() throws IOException;
}

BufferedInputStream是InputStream子类的子类,在继承关系上,BufferedInputStream继承FilterInputStream,FilterInputStream继承InputStream,这里相当于对装饰类进行了再扩展,看一下FilterInputStream:

public class FilterInputStream extends InputStream{
    protected volatile InputStream in;
    protected FilterInputStream(InputStream in) {
        this.in = in;
    }
    public int read() throws IOException {
        return in.read();
    }
}

而BufferedInputStream以及DataInputStream都是对FilterInputStream再做一些功能上的增强,很巧妙的实现了在不必改变原类文件情况下,允许向一个现有的对象添加新的功能。

总结

装饰者模式很好地体现了设计模式中的开闭原则,即类应该对扩展开放,对修改关闭。值得深深品味。

暂无评论

发送评论 编辑评论


				
|´・ω・)ノ
ヾ(≧∇≦*)ゝ
(☆ω☆)
(╯‵□′)╯︵┴─┴
 ̄﹃ ̄
(/ω\)
∠( ᐛ 」∠)_
(๑•̀ㅁ•́ฅ)
→_→
୧(๑•̀⌄•́๑)૭
٩(ˊᗜˋ*)و
(ノ°ο°)ノ
(´இ皿இ`)
⌇●﹏●⌇
(ฅ´ω`ฅ)
(╯°A°)╯︵○○○
φ( ̄∇ ̄o)
ヾ(´・ ・`。)ノ"
( ง ᵒ̌皿ᵒ̌)ง⁼³₌₃
(ó﹏ò。)
Σ(っ °Д °;)っ
( ,,´・ω・)ノ"(´っω・`。)
╮(╯▽╰)╭
o(*////▽////*)q
>﹏<
( ๑´•ω•) "(ㆆᴗㆆ)
😂
😀
😅
😊
🙂
🙃
😌
😍
😘
😜
😝
😏
😒
🙄
😳
😡
😔
😫
😱
😭
💩
👻
🙌
🖕
👍
👫
👬
👭
🌚
🌝
🙈
💊
😶
🙏
🍦
🍉
😣
Source: github.com/k4yt3x/flowerhd
颜文字
Emoji
小恐龙
花!
上一篇
下一篇