java的设计模式之桥接模式(Bridge)

之前遇到过一个采用发送邮件和发送短信的方式低级中级高级三种告警通知,当时做的很潦草,现在发现了大佬的文章桥接模式,我感觉很好,非常适用那个场景,于是想发出来,各位都看看

桥接模式包含以下四个关键角色:

抽象化: 定义抽象类的接口,并维护一个指向实现化对象的引用。

扩展抽象化: 抽象化的子类,可以扩展或修改父类的接口。

实现化: 定义实现类的接口。这个接口不一定要与抽象化的接口完全一致,通常更基础。

具体实现化: 实现化接口的具体类。

java 复制代码
//通知 抽象类
abstract class Notification {

    protected MessageSender sender;

    public Notification(MessageSender sender) {
        this.sender = sender;
    }
    public abstract void notify(String message);
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 *  扩展的抽象 低级告警
 * @author guanc
 * @ClassName LowLevelNotification
 * @date 2025年11月07日
 * @version: 1.0
 */
public class LowLevelNotification extends Notification{

    public LowLevelNotification(MessageSender sender) {
        super(sender);
    }

    @Override
    public void notify(String message) {
        sender.send( message);
    }
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 *  扩展的抽象 中级告警
 * @author guanc
 * @ClassName LowLevelNotification
 * @date 2025年11月07日
 * @version: 1.0
 */
public class IntermediateNotification extends Notification{

    public IntermediateNotification(MessageSender sender) {
        super(sender);
    }

    @Override
    public void notify(String message) {
        //重试逻辑 尝试发送3次
        int retries = 3;
        while (retries > 0){
            try {
                sender.send( message);
                break;
            } catch (Exception e) {
                retries--;
            }
        }
    }
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 *  扩展的抽象 高级
 * @author guanc
 * @ClassName LowLevelNotification
 * @date 2025年11月07日
 * @version: 1.0
 */
public class AdvancedNotification extends Notification{

    private long delayInSecond;

    public AdvancedNotification(MessageSender sender,long delayInSecond) {
        super(sender);
        this.delayInSecond = delayInSecond;
    }

    @Override
    public void notify(String message) {
       new Thread(() -> {
           try {
               Thread.sleep(delayInSecond * 1000);
               sender.send( message);
           } catch (InterruptedException e) {
               e.printStackTrace();
           }
       }).start();
        System.out.println("延迟了" + delayInSecond + "秒后发送");
    }
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 * 实现的接口 消息发送器
 * @author guanc
 * @ClassName MessageSender
 * @date 2025年11月07日
 * @version: 1.0
 */
interface MessageSender {
   void send(String message);
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 * 短信发送器
 * @author guanc
 * @ClassName EmailSender
 * @date 2025年11月07日
 * @version: 1.0
 */
public class SmsSender implements MessageSender{
    @Override
    public void send(String message) {
        System.out.println("短信发送:" + message);
    }
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 * 短信发送器
 * @author guanc
 * @ClassName EmailSender
 * @date 2025年11月07日
 * @version: 1.0
 */
public class EmailSender implements MessageSender{
    @Override
    public void send(String message) {
        System.out.println("发送邮件:" + message);
    }
}
java 复制代码
package com.sailing.hsm.bridge;

/**
 *
 * @author guanc
 * @ClassName TestDemo
 * @date 2025年11月07日
 * @version: 1.0
 */
public class TestDemo {
    public static void main(String[] args) {
        MessageSender emailSender = new EmailSender();
        MessageSender smsSender = new SmsSender();
        //中级告警+短信
        Notification intermediateNotification = new IntermediateNotification(smsSender);
        intermediateNotification .notify("内存占用百分之90了,少年还不来看看!!!");
        //低级告警+邮件
        Notification lowLevelNotification = new LowLevelNotification(emailSender);
        LowLevelNotification.notify(" 内存占用百分之80,少年来看看吧!");
        //高级告警+短信
        Notification advancedNotification = new AdvancedNotification(smsSender,6);
        advancedNotification .notify("服务器爆炸了,赶紧修复啊!!!" );
    }
}

它的适用场景

  • 当一个类存在两个独立变化的维度,且这两个维度都需要进行扩展时。

  • 当一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性时。

  • 不希望使用继承或因为多层次继承导致系统类的个数急剧增加时。

  • 需要对抽象部分和实现部分进行动态绑定时。

咋一看桥接模式与适配器模式有点类似,但是你仔细看会发现很大不同:

桥接模式: 在设计之初就使用,目的是将抽象和实现分离,以便它们可以独立变化。它是一种前瞻性的设计。

适配器模式: 通常在设计之后使用,目的是让两个原本不兼容的接口能够协同工作。它是一种补救性的设计。

相关推荐
二哈赛车手1 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
吃好睡好便好2 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
栗子~~2 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8292 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
仰泳之鹅2 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
未若君雅裁3 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
AI人工智能+电脑小能手4 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
阿维的博客日记4 小时前
Nacos 为什么能让配置动态生效?(涉及 @RefreshScope 注解)
java·spring
雨辰AI4 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务
x_yeyue4 小时前
三角形数
笔记·算法·数论·组合数学