Java设计模式——责任链模式

当一个请求需要在多个对象之间传递,每个对象都可能处理该请求或将其传递给下一个对象。在这种情况下,需要避免将发送者与接收者之间的耦合,以及确定请求的处理方式。此时可使用责任链模式,它的优点有降低耦合度(无需关心哪个对象会处理请求),灵活性(可以动态地改变它们之间的顺序和职责),可扩展性(添加新的处理者,而不会影响现有代码),可维护性(处理者之间职责分开,便于维护);责任链模式也有限制,比如可能导致请求无法被处理或者处理链太长而导致性能问题。

一、抽象类方式实现

假定一个权限认证的例子

java 复制代码
/**
 * 成员类
 */
@Getter
@ToString
public class Member {
    private String loginName;
    private String loginPass;
    @Setter
    private String roleName;

    public Member(String loginName, String loginPass) {
        this.loginName = loginName;
        this.loginPass = loginPass;
    }
}


/**
 * 抽象处理者
 */
public abstract class Handler {
    protected Handler handler;

    public void next(Handler handler) {
        this.handler = handler;
    }

    public abstract void doHandler(Member member);
}


/**
 * 认证处理器
 */
public class AuthHandler extends Handler {
    @Override
    public void doHandler(Member member) {
        if (!"管理员".equals(member.getRoleName())) {
            System.out.println("无操作权限!");
            return;
        }
        System.out.println("操作成功,管理员同志");
    }
}


/**
 * 登陆处理器
 */
public class LoginHandler extends Handler {
    @Override
    public void doHandler(Member member) {
        System.out.println("登陆成功!");
        member.setRoleName("管理员");
        handler.doHandler(member);
    }
}


/**
 * 校验数据处理器
 */
public class ValidDataHandler extends Handler {
    @Override
    public void doHandler(Member member) {
        if (StringUtils.isBlank(member.getLoginName()) || StringUtils.isBlank(member.getLoginPass())) {
            System.out.println("用户名或密码不能为空!");
            return;
        }
        System.out.println("验证通过!");
        handler.doHandler(member);
    }
}


/**
 * 业务服务类
 */
public class MemberService {

    public void login(String loginName, String loginPass) {
        Handler validDateHandler = new ValidDataHandler();
        Handler loginHandler = new LoginHandler();
        Handler authHandler = new AuthHandler();

        validDateHandler.next(loginHandler);
        loginHandler.next(authHandler);

        validDateHandler.doHandler(new Member(loginName, loginPass));
    }

}

编写测试用例

java 复制代码
    /**
     * 责任链模式-权限认证测试
     */
    @Test
    public void testPermission() {
        MemberService memberService = new MemberService();
        memberService.login("admin", "123456");
    }

二、与建造者模式相结合

还是权限认证的例子

java 复制代码
/**
 * 抽象处理者(结合建造者模式)
 */
public abstract class HandlerUp {
    protected HandlerUp handler;

    public void next(HandlerUp handler) {
        this.handler = handler;
    }

    public abstract void doHandler(Member member);

    public static class Builder {
        private HandlerUp head;
        private HandlerUp tail;

        public Builder addHandler(HandlerUp handler) {
            if (this.head == null) {
                this.head = this.tail = handler;
                return this;
            }
            this.tail.next(handler);
            this.tail = handler;

            return this;
        }

        public HandlerUp build() {
            return this.head;
        }
    }
}


/**
 * 认证处理器
 */
public class AuthHandlerUp extends HandlerUp {
    @Override
    public void doHandler(Member member) {
        if (!"管理员".equals(member.getRoleName())) {
            System.out.println("非管理员,无操作权限!");
            return;
        }
        System.out.println("操作成功,管理员同志");
    }
}


/**
 * 登陆处理器
 */
public class LoginHandlerUp extends HandlerUp {
    @Override
    public void doHandler(Member member) {
        System.out.println("登陆成功!");
        member.setRoleName("管理员");
        handler.doHandler(member);
    }
}


/**
 * 校验数据处理器
 */
public class ValidDataHandlerUp extends HandlerUp {
    @Override
    public void doHandler(Member member) {
        if (StringUtils.isBlank(member.getLoginName()) || StringUtils.isBlank(member.getLoginPass())) {
            System.out.println("用户名或密码不能为空!");
            return;
        }
        System.out.println("用户名和密码校验成功!");
        handler.doHandler(member);
    }
}


/**
 * 业务服务类
 */
public class MemberServiceUp {

    public void login(String loginName, String loginPass) {
        new HandlerUp.Builder()
                .addHandler(new ValidDataHandlerUp())
                .addHandler(new LoginHandlerUp())
                .addHandler(new AuthHandlerUp())
                .build()
                .doHandler(new Member(loginName, loginPass));
    }

}

编写测试用例

java 复制代码
    /**
     * 责任链模式-权限认证(增强版)测试
     */
    @Test
    public void testPermissionUp() {
        MemberServiceUp memberServiceUp = new MemberServiceUp();
        memberServiceUp.login("admin", "123456");
    }
相关推荐
庄周的大鱼35 分钟前
分析@TransactionalEventListener注解失效
java·spring·springboot·事务监听器·spring 事件机制·事务注解失效解决
史蒂芬_丁1 小时前
C++深度拷贝例子
java·开发语言·c++
云烟成雨TD1 小时前
Spring AI Alibaba 1.x 系列【4】ReAct 范式与 ReactAgent 核心设计
java·人工智能·spring
「QT(C++)开发工程师」2 小时前
C++11三大核心特性深度解析:类型特征、时间库与原子操作
java·c++·算法
乐分启航2 小时前
SliMamba:十余K参数量刷新SOTA!高光谱分类的“降维打击“来了
java·人工智能·深度学习·算法·机器学习·分类·数据挖掘
yoothey3 小时前
Java字节流与字符流核心笔记(问答+考点复盘)
java·开发语言·笔记
black方块cxy3 小时前
实现一个输入框多个ip以逗号分隔最多20组,且ip不能重复
java·服务器·前端
23.4 小时前
【Java】char字符类型的UTF-16编码解析
java·开发语言·面试
怒放吧德德4 小时前
Spring Boot实战:InfluxDB 2.x简单教程
java·spring boot·后端
indexsunny4 小时前
互联网大厂Java面试实战:核心技术与业务场景深度解析
java·spring boot·hibernate·security·microservices·interview