组合模式的例子

news2024/9/29 9:32:39
// 组合模式的接口
public interface AccessDecisionVoter {
    // 投票结果的常量
    int ACCESS_GRANTED = 1;
    int ACCESS_ABSTAIN = 0;
    int ACCESS_DENIED = -1;
    // 投票方法,根据用户和请求判断是否授权
    int vote(User user, Request request);
}

// 组合模式的叶子节点,实现了AccessDecisionVoter接口
// 这个类用于判断用户是否有角色权限
public class RoleVoter implements AccessDecisionVoter {
    @Override
    public int vote(User user, Request request) {
        // 获取用户的角色列表
        List<String> roles = user.getRoles();
        // 获取请求需要的角色列表
        List<String> requiredRoles = request.getRequiredRoles();
        // 如果用户的角色列表包含请求需要的任意一个角色,则授权通过
        for (String role : requiredRoles) {
            if (roles.contains(role)) {
                return ACCESS_GRANTED;
            }
        }
        // 否则授权拒绝
        return ACCESS_DENIED;
    }
}

// 组合模式的叶子节点,实现了AccessDecisionVoter接口
// 这个类用于判断用户是否有资源权限
public class ResourceVoter implements AccessDecisionVoter {
    @Override
    public int vote(User user, Request request) {
        // 获取用户的资源列表
        List<String> resources = user.getResources();
        // 获取请求需要的资源列表
        List<String> requiredResources = request.getRequiredResources();
        // 如果用户的资源列表包含请求需要的所有资源,则授权通过
        for (String resource : requiredResources) {
            if (!resources.contains(resource)) {
                return ACCESS_DENIED;
            }
        }
        return ACCESS_GRANTED;
    }
}

// 组合模式的容器节点,实现了AccessDecisionVoter接口
// 这个类用于组合多个AccessDecisionVoter对象,并根据投票结果决定是否授权
public class AccessDecisionManager implements AccessDecisionVoter {
    // 组合多个AccessDecisionVoter对象,使用List存储
    private List<AccessDecisionVoter> voters;

    public AccessDecisionManager(List<AccessDecisionVoter> voters) {
        this.voters = voters;
    }

    @Override
    public int vote(User user, Request request) {
        // 记录投票结果的变量
        int result = ACCESS_ABSTAIN;
        // 遍历所有的AccessDecisionVoter对象,进行投票
        for (AccessDecisionVoter voter : voters) {
            int vote = voter.vote(user, request);
            // 如果有任何一个投票结果是拒绝,则直接返回拒绝
            if (vote == ACCESS_DENIED) {
                return ACCESS_DENIED;
            }
            // 如果有任何一个投票结果是通过,则记录为通过,但不返回,继续遍历其他投票者
            if (vote == ACCESS_GRANTED) {
                result = ACCESS_GRANTED;
            }
        }
        // 返回最终的投票结果
        return result;
    }
}

// 用户类,用于模拟用户信息,包括用户名、密码、角色列表和资源列表
public class User {
    private String username;
    private String password;
    private List<String> roles;
    private List<String> resources;

    public User(String username, String password, List<String> roles, List<String> resources) {
        this.username = username;
        this.password = password;
        this.roles = roles;
        this.resources = resources;
    }

    public String getUsername() {
        return username;
    }

    public String getPassword() {
        return password;
    }

    public List<String> getRoles() {
        return roles;
    }

    public List<String> getResources() {
        return resources;
    }
}

// 请求类,用于模拟请求信息,包括请求路径、请求方法、请求需要的角色列表和资源列表
public class Request {
    private String path;
    private String method;
    private List<String> requiredRoles;
    private List<String> requiredResources;

    public Request(String path, String method, List<String> requiredRoles, List<String> requiredResources) {
        this.path = path;
        this.method = method;
        this.requiredRoles = requiredRoles;
        this.requiredResources = requiredResources;
    }

    public String getPath() {
        return path;
    }

    public String getMethod() {
        return method;
    }

    public List<String> getRequiredRoles() {
        return requiredRoles;
    }

    public List<String> getRequiredResources() {
        return requiredResources;
    }
}

// 依赖注入的类,用于模拟从数据库中获取用户信息和请求信息
// 这里使用了@Component注解,让Spring容器管理这个类的实例
@Component
public class DataProvider {
    // 模拟数据库中的用户信息,使用Map存储
    private Map<String, User> users;

    // 模拟数据库中的请求信息,使用Map存储
    private Map<String, Request> requests;

    // 在构造方法中初始化数据
    public DataProvider() {
        users = new HashMap<>();
        requests = new HashMap<>();
        // 创建用户对象,并添加到Map中
        User user1 = new User("admin", "123456", Arrays.asList("admin", "user"), Arrays.asList("read", "write", "delete"));
        User user2 = new User("user1", "123456", Arrays.asList("user"), Arrays.asList("read", "write"));
        User user3 = new User("user2", "123456", Arrays.asList("user"), Arrays.asList("read"));
        users.put(user1.getUsername(), user1);
        users.put(user2.getUsername(), user2);
        users.put(user3.getUsername(), user3);
        // 创建请求对象,并添加到Map中
        Request request1 = new Request("/api/user/list", "GET", Arrays.asList("admin"), Arrays.asList("read"));
        Request request2 = new Request("/api/user/add", "POST", Arrays.asList("admin"), Arrays.asList("write"));
        Request request3 = new Request("/api/user/delete", "DELETE", Arrays.asList("admin"), Arrays.asList("delete"));
        Request request4 = new Request("/api/user/update", "PUT", Arrays.asList("user"), Arrays.asList("write"));
        requests.put(request1.getPath(), request1);
        requests.put(request2.getPath(), request2);
        requests.put(request3.getPath(), request3);
        requests.put(request4.getPath(), request4);
    }

    // 根据用户名获取用户对象
    public User getUserByUsername(String username) {
        return users.get(username);
    }

    // 根据请求路径获取请求对象
    public Request getRequestByPath(String path) {
        return requests.get(path);
    }
}

// 控制器类,用于模拟接收用户的请求,并调用AccessDecisionManager进行权限验证
// 这里使用了@RestController注解,让Spring容器管理这个类的实例,并将返回值转换为JSON格式
@RestController
public class UserController {
    // 使用@Autowired注解,让Spring容器自动注入DataProvider实例
    @Autowired
    private DataProvider dataProvider;

    // 使用@Autowired注解,让Spring容器自动注入AccessDecisionManager实例
    @Autowired
    private AccessDecisionManager accessDecisionManager;

    // 使用@RequestMapping注解,映射请求路径和方法
    @RequestMapping(value = "/api/user/list", method = RequestMethod.GET)
    public String listUsers(String username, String password) {
        // 模拟从请求中获取用户名和密码
        // 根据用户名从DataProvider中获取用户对象
        User user = dataProvider.getUserByUsername(username);
        // 如果用户不存在或密码不正确,返回错误信息
        if (user == null || !user.getPassword().equals(password)) {
            return "用户名或密码错误";
        }
        // 根据请求路径从DataProvider中获取请求对象
        Request request = dataProvider.getRequestByPath("/api/user/list");
        // 调用AccessDecisionManager的vote方法进行权限验证
        int result = accessDecisionManager.vote(user, request);
        // 如果结果是通过,则返回成功信息
        if (result == AccessDecisionVoter.ACCESS_GRANTED) {
            return "查询用户列表成功";
        }
        // 否则返回失败信息
        return "没有权限查询用户列表";
    }

    @RequestMapping(value = "/api/user/add", method = RequestMethod.POST)
    public String addUser(String username, String password) {
       // 省略与上面相同的代码

       // 根据请求路径从DataProvider中获取请求
        Request request = dataProvider.getRequestByPath("/api/user/add");
        // 调用AccessDecisionManager的vote方法进行权限验证
        int result = accessDecisionManager.vote(user, request);
        // 如果结果是通过,则返回成功信息
        if (result == AccessDecisionVoter.ACCESS_GRANTED) {
            return "添加用户成功";
        }
        // 否则返回失败信息
        return "没有权限添加用户";
    }

    @RequestMapping(value = "/api/user/delete", method = RequestMethod.DELETE)
    public String deleteUser(String username, String password) {
       // 省略与上面相同的代码

       // 根据请求路径从DataProvider中获取请求对象
        Request request = dataProvider.getRequestByPath("/api/user/delete");
        // 调用AccessDecisionManager的vote方法进行权限验证
        int result = accessDecisionManager.vote(user, request);
        // 如果结果是通过,则返回成功信息
        if (result == AccessDecisionVoter.ACCESS_GRANTED) {
            return "删除用户成功";
        }
        // 否则返回失败信息
        return "没有权限删除用户";
    }

    @RequestMapping(value = "/api/user/update", method = RequestMethod.PUT)
    public String updateUser(String username, String password) {
       // 省略与上面相同的代码

       // 根据请求路径从DataProvider中获取请求对象
        Request request = dataProvider.getRequestByPath("/api/user/update");
        // 调用AccessDecisionManager的vote方法进行权限验证
        int result = accessDecisionManager.vote(user, request);
        // 如果结果是通过,则返回成功信息
        if (result == AccessDecisionVoter.ACCESS_GRANTED) {
            return "更新用户成功";
        }
        // 否则返回失败信息
        return "没有权限更新用户";
    }
}

组合模式是一种结构型设计模式,它可以将对象组合成树形结构,表示“整体-部分”的层次关系。组合模式可以让客户端以一致的方式处理单个对象和组合对象。

在Java中,有一些知名的框架使用了组合模式,例如:

Swing框架中的JComponent类和其子类,如JPanel, JButton, JLabel等,都是组合模式的例子。JComponent是抽象构件类,JPanel是容器构件类,JButton和JLabel是叶子构件类。
Spring框架中的BeanDefinition类和其子类,如RootBeanDefinition, GenericBeanDefinition等,也是组合模式的例子。BeanDefinition是抽象构件类,RootBeanDefinition是容器构件类,GenericBeanDefinition是叶子构件类。
MyBatis框架中的SqlNode接口和其实现类,如MixedSqlNode, IfSqlNode, TextSqlNode等,也是组合模式的例子。SqlNode是抽象构件接口,MixedSqlNode是容器构件类,IfSqlNode和TextSqlNode是叶子构件类。

一个简单入门的小例子:

// 抽象构件类,表示文件或文件夹
public abstract class File {
    // 文件或文件夹的名字
    protected String name;
    // 构造方法,传入名字
    public File(String name) {
        this.name = name;
    }
    // 公共的业务方法,显示文件或文件夹的信息
    public abstract void display();
    // 添加子文件或子文件夹
    public void add(File file) {
        throw new UnsupportedOperationException();
    }
    // 删除子文件或子文件夹
    public void remove(File file) {
        throw new UnsupportedOperationException();
    }
    // 获取子文件或子文件夹
    public File getChild(int i) {
        throw new UnsupportedOperationException();
    }
}

// 叶子构件类,表示文本文件
public class TextFile extends File {
    // 构造方法,传入文本文件的名字
    public TextFile(String name) {
        super(name);
    }
    // 重写业务方法,输出文本文件的信息
    @Override
    public void display() {
        System.out.println("TextFile: " + name);
    }
}

// 叶子构件类,表示图片文件
public class ImageFile extends File {
    // 构造方法,传入图片文件的名字
    public ImageFile(String name) {
        super(name);
    }
    // 重写业务方法,输出图片文件的信息
    @Override
    public void display() {
        System.out.println("ImageFile: " + name);
    }
}

// 容器构件类,表示文件夹
public class Folder extends File {
    // 存储子文件或子文件夹的列表
    private List<File> children;
    // 构造方法,传入文件夹的名字
    public Folder(String name) {
        super(name);
        children = new ArrayList<>();
    }
    // 重写业务方法,输出文件夹的信息,并递归调用子文件或子文件夹的业务方法
    @Override
    public void display() {
        System.out.println("Folder: " + name);
        for (File child : children) {
            child.display();
        }
    }
    // 重写添加子文件或子文件夹方法,将子文件或子文件夹添加到列表中
    @Override
    public void add(File file) {
        children.add(file);
    }
    // 重写删除子文件或子文件夹方法,将子文件或子文件夹从列表中移除
    @Override
    public void remove(File file) {
        children.remove(file);
    }
    // 重写获取子文件或子文件夹方法,根据索引返回对应的子文件或子文件夹
    @Override
    public File getChild(int i) {
        return children.get(i);
    }
}

// 客户端测试类
public class Client {
    public static void main(String[] args) {
        // 创建一个根文件夹
        Folder root = new Folder("root");
        // 创建两个文本文件和一个图片文件,并添加到根文件夹中
        TextFile text1 = new TextFile("text1.txt");
        TextFile text2 = new TextFile("text2.txt");
        ImageFile image1 = new ImageFile("image1.jpg");
        root.add(text1);
        root.add(text2);
        root.add(image1);
        // 创建一个子文件夹,并添加到根文件夹中
        Folder subFolder = new Folder("subFolder");
        root.add(subFolder);
        // 创建一个文本文件和一个图片文件,并添加到子文件夹中
        TextFile text3 = new TextFile("text3.txt");
        ImageFile image2 = new ImageFile("image2.jpg");
        subFolder.add(text3);
        subFolder.add(image2);
        // 调用根文件夹的业务方法,输出整个树形结构
        root.display();
        
         /*
         * 输出结果:
         * Folder: root
         * TextFile: text1.txt
         * TextFile: text2.txt
         * ImageFile: image1.jpg
         * Folder: subFolder
         * TextFile: text3.txt
         * ImageFile: image2.jpg
         */
        
    }
}

合成模式:
1.安全模式
2.透明模式
安全模式:
这里写图片描述
透明模式
这里写图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/737533.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Android中system/bin/Input命令 -- Android12

IMS:Android中Input命令--Android12 1、Android12 Input命令更新1.1 shell脚本1.2 InputShellCommand#onCommand 命令解析 2、Input相关命令参数2.1 text2.2 keyevent2.3 tap2.4 swipe2.5 draganddrop2.6 press2.7 roll2.8 motionevent2.9 keycombination2.10 默认handleDefaul…

2023 WAIC图技术激活数据要素论坛圆满召开!

7月6日&#xff0c;以“智联世界 生成未来”为主题的2023世界人工智能大会&#xff08;WAIC 2023&#xff09;在上海隆重开幕。作为大会唯一的图技术论坛&#xff0c;“图技术激活数据要素论坛”也如期举行。 论坛现场&#xff0c;学术界专家学者、头部银行代表、产业界大咖齐聚…

二分图博弈(知识总结+例题)

思路来源 gzchenben的ppt 算法学习笔记(74): 二分图博弈 - 知乎 https://www.cnblogs.com/Zeardoe/p/16534557.html 知识点总结 以下部分摘自知乎&#xff1a;算法学习笔记(74): 二分图博弈 - 知乎 二分图博弈模型 给出一张二分图和起始点 H &#xff0c; A和B轮流操作…

Endnotes引用

准备不用zetro了&#xff0c;主要学校endnotes免费 该说不说&#xff0c;zetro拖入pdf直接识别并导入的功能是真的好用&#xff0c;添加备注也方便 可惜文献太多放不下了&#xff0c;扩容要加钱。 啧。算了算了。 这里主要介绍Endnotes中的文献怎么在word内引用&#xff0c…

SpringBoot 插件化开发模式,强烈推荐!

一、前言 插件化开发模式正在很多编程语言或技术框架中得以广泛的应用实践&#xff0c;比如大家熟悉的jenkins&#xff0c;docker可视化管理平台rancher&#xff0c;以及日常编码使用的编辑器idea&#xff0c;vscode等&#xff0c;随处可见的带有热插拔功能的插件&#xff0c;…

BitLocker 驱动器加密管理

为了有效地保护数字数据&#xff0c;应对其进行加密&#xff0c;以便只有授权用户才能访问。BitLocker 是某些 Windows 操作系统上可用的本机加密工具&#xff0c;可以为个人用户轻松加密Windows计算机。 什么是 BitLocker 加密 BitLocker 加密是 Windows 操作系统的内置安全…

STM32实现气压传感器测量(BMP180)

目录 0.接线设计 1.功能描述 2.四种方式实现大气压采集 3.模块选择 4.编程环境 5.模块主要参数 6.代码实现 1&#xff09;标准库模拟IIC实现气压值采集 2&#xff09;标准库硬件IIC实现气压值采集 3&#xff09;HAL库模拟IIC实现气压值采集 4&#xff09;HAL库硬件IIC实…

分享一次腾讯云轻量应用服务器被攻击

腾讯云轻量应用服务器&#xff0c;centOS。在上面装了redis、rabbit mq等服务&#xff0c;开着端口&#xff0c;结果被入侵了。 发现问题是通过腾讯云发来的邮件&#xff0c; 首先进到主机安全控制台&#xff0c;左侧这里进主机列表 然后可以看到自己的主机情况&#xff0c;防…

ROS:机器人系统仿真

目录 一、概念二、作用2.1仿真优势:2.2仿真缺陷: 三、组件3.1URDF3.2rviz3.3gazebo 一、概念 通过计算机对实体机器人系统进行模拟的技术&#xff0c;在 ROS 中&#xff0c;仿真实现涉及的内容主要有三:对机器人建模(URDF)、创建仿真环境(Gazebo)以及感知环境(Rviz)等系统性实…

HTML特性(attribute)和DOM属性(property)

文章目录 定义位置不同attributeproperty 范围不同属性映射行为区别数据类型不同大小写敏感区别相同属性返回值可能不同DOM 属性具有写保护 定义位置不同 attribute 是 HTML 标签上的某个属性&#xff0c;如 id、class、value 等以及自定义属性,定义后会呈现在标签上 proper…

赛效:如何在线编辑图片

1&#xff1a;点击导航栏里的“图片编辑”。 2&#xff1a;点击打开图片或者拖放打开图片。 3&#xff1a;左侧几十种工具&#xff0c;你可以用来在线编辑图片。 4&#xff1a;编辑完成后点击页面右上角的“下载”按钮&#xff0c;根据提示登录账号下载图片就可以了。 如果你想…

2023广州建博会:鸿雁总裁王米成详解全屋智能的发展脉络

全屋智能落地的模式有很多&#xff0c;但鸿雁依托其智能面板优势&#xff0c;逐渐探索出一条属于鸿雁的全屋智能发展路径和商业模式。 智哪儿创始人、总编彭安军&#xff08;左&#xff09;&#xff0c;鸿雁电器总裁王米成&#xff08;右&#xff09; 在2023年的广州建博会上&a…

【Linux从入门到放弃】进程状态的理解以及什么是僵尸进程和孤儿进程?

&#x1f9d1;‍&#x1f4bb;作者&#xff1a; 情话0.0 &#x1f4dd;专栏&#xff1a;《Linux从入门到放弃》 &#x1f466;个人简介&#xff1a;一名双非编程菜鸟&#xff0c;在这里分享自己的编程学习笔记&#xff0c;欢迎大家的指正与点赞&#xff0c;谢谢&#xff01; 文…

你知道什么是基于StyleNeRF的conditional GAN模型吗

随着深度学习技术的不断发展&#xff0c;生成对抗网络&#xff08;GAN&#xff09;已经成为了人工智能研究和应用中的重要组成部分。其中&#xff0c;GAN可以被用来生成高质量的图像、视频等内容&#xff0c;这为娱乐产业和数字化制作带来了新的机遇和挑战。本文将介绍一种基于…

numpy 笔记 pad

1 基本介绍 对原本的矩阵进行填充 numpy.pad(array, pad_width, modeconstant, **kwargs) array待填充的矩阵pad_width要填充的位置mode填充方式 2 参数举例 2.0 使用的数据 import numpy as np anp.arange(12).reshape(3,4) aarray([[ 0, 1, 2, 3],[ 4, 5, 6, 7],…

【分布式 】 ELK 企业级日志分析系统

目录 一、ELK概述1.1 ELK简介1.2 为什么要使用ELK1.3 完整日志系统基本特征1.4 ELK的工作原理 二、搭建ELK2.1 ELK Elasticsearch 集群部署&#xff08;在Node1、Node2节点上操作&#xff09;node1 操作node2 同node1操作 2.2 安装 Elasticsearch-head 插件ELK Logstash 部署&a…

作为一个测试工程师,你选择python还是java?

问&#xff1a;“你平时工作中&#xff0c;用java多还是用python多”&#xff1f; 答&#xff1a;“都还可以&#xff0c;根据具体的场景选择不同的语言”。 问&#xff1a;“比如说呢”&#xff1f; 答&#xff1a;“开发自己的测试平台&#xff0c;肯定会选择java&#xf…

Stable Diffusion - ControlNet 插件中扩展局部重绘 InpaintOnly + LaMa 的算法与应用

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/131643131 LaMa: https://github.com/advimman/lama Paper: Resolution-robust Large Mask Inpainting with Fourier Convolutions LaMa: Large…

ActiveMQ详细入门教程系列

一、什么是消息中间件 两个系统或两个客户端之间进行消息传送&#xff0c;利用高效可靠的消息传递机制进行平台无关的数据交流&#xff0c;并基于数据通信来进行分布式系统的集成。通过提供消息传递和消息排队模型&#xff0c;它可以在分布式环境下扩展进程间的通信。 消息中…

ABeam中国2023社招 | ABeam旗下德硕管理咨询(深圳)招贤纳士

岗位需求 SAP Basis顾问 岗位职责 ■ 参与公司的SAP售前项目&#xff0c;负责Basis相关工作的方案制定 ■ 参与公司既有SAP运维项目&#xff0c;负责Basis相关的课题对应&#xff0c;系统改善等 ■ 负责SAP系统的Basis实施&#xff0c;SAP产品系统安装、升级、迁移、数据归档…