Java桥接模式源码剖析及使用场景

news/2024/7/5 2:45:30

目录

  • 一、介绍
  • 二、项目管理系统中使用桥接模式
  • 三、权限管理中使用桥接模式
  • 四、Java JDBC中使用桥接模式

一、介绍

它的主要目的是将抽象化与实现化分离,使得二者可以独立变化,就像一个桥,将两个变化维度连接起来。各个维度都可以独立的变化。故称之为:桥接模式

桥接模式的核心在于通过一个桥接接口,将抽象部分与实现部分解耦。这样做的好处是显而易见的,当系统中的某个维度(抽象或实现)需要变更时,不会影响到另一个维度。具体来说,桥接模式涉及以下几个关键角色:

  1. 抽象化角色(Abstraction):它是抽象类的接口,定义了一个实现化的引用和对实现化的操作。

  2. 修正抽象化角色(Refined Abstraction):它是扩展了抽象化角色的子类,在实际应用中,具体的业务逻辑通常在这个角色中实现。

  3. 实现化角色(Implementor):这个角色定义了一个接口,由具体实现化角色来实现。

  4. 具体实现化角色(Concrete Implementor):这个角色实现了实现化接口的具体类。

二、项目管理系统中使用桥接模式

需求:项目管理系统,包含多种任务(编码任务,测试任务)和多种任务执行(本地执行,远程执行)方式 。还有开发和测试角色。我们可以使用桥接模式来分离任务和任务执行方式的实现,以便它们可以独立变化。

  • 定义项目管理系统中的任务接口
public interface Task {
    void performTask();
}
  • 具体的任务实现类
// 编码任务
public class CodingTask implements Task {
    @Override
    public void performTask() {
        System.out.println("Performing coding task");
    }
}

// 测试任务
public class TestingTask implements Task {
    @Override
    public void performTask() {
        System.out.println("Performing testing task");
    }
}
  • 任务执行方式接口
public interface TaskExecution {
    void executeTask();
}
  • 具体的任务执行方式实现类
// 本地执行任务
public class LocalTaskExecution implements TaskExecution {
    @Override
    public void executeTask() {
        System.out.println("Executing task locally");
    }
}

// 远程执行任务
public class RemoteTaskExecution implements TaskExecution {
    @Override
    public void executeTask() {
        System.out.println("Executing task remotely");
    }

  • 项目管理系统角色抽象类
public abstract class ProjectRole {
    protected Task task;
    protected TaskExecution taskExecution;

    public void setTask(Task task) {
        this.task = task;
    }

    public void setTaskExecution(TaskExecution taskExecution) {
        this.taskExecution = taskExecution;
    }

    public abstract void performProjectTask();
}
  • 项目管理系统具体角色实现类
// 开发人员角色
public class Developer extends ProjectRole {
    @Override
    public void performProjectTask() {
        System.out.print("Developer role: ");
        task.performTask();
        System.out.print("Using ");
        taskExecution.executeTask();
    }
}

// 测试人员角色
public class Tester extends ProjectRole {
    @Override
    public void performProjectTask() {
        System.out.print("Tester role: ");
        task.performTask();
        System.out.print("Using ");
        taskExecution.executeTask();
    }
}
  • 客户端代码演示桥接模式的使用
public class BridgePatternDemo {
    public static void main(String[] args) {
        Task codingTask = new CodingTask();
        Task testingTask = new TestingTask();
        TaskExecution localTaskExecution = new LocalTaskExecution();
        TaskExecution remoteTaskExecution = new RemoteTaskExecution();

        // 创建不同角色的实例
        ProjectRole developer = new Developer();
        ProjectRole tester = new Tester();

        // 设置不同的任务和任务执行方式
        developer.setTask(codingTask);
        developer.setTaskExecution(remoteTaskExecution);

        tester.setTask(testingTask);
        tester.setTaskExecution(localTaskExecution);

        // 视图数据
        developer.performProjectTask();  // Output: "Developer role: Performing coding task Using Executing task remotely"
        tester.performProjectTask();   // Output: "Tester role: Performing testing task Using Executing task locally"
    }
}

大家说上面七步是桥接模式吗?在实际项目中,我们可以根据需要添加更多的任务和任务执行方式,而不需要修改已有的代码。核心在于正确识别并分离出两个独立的维度,并且这两个维度应该能够各自独立地变化和扩展。

三、权限管理中使用桥接模式

需求:权限管理系统,分别有普通角色、管理员角色、超级管理员角色。以数据权限设计,普通角色只能看到自己操作的数据,管理员角色可以看到普通角色操作的数据,超级管理员角色可以看到所有角色的数据。

  • 定义数据权限控制策略接口
public interface DataPermissionControlStrategy {
    void process();
}
  • 定义具体的数据权限控制策略实现类
// 普通角色只能看到自己操作的数据
public class NormalDataPermissionControlStrategy implements DataPermissionControlStrategy {
    @Override
    public void process() {
        System.out.println("Normal role can only access own data");
    }
}

// 管理员角色可以看到普通角色操作的数据
public class AdminDataPermissionControlStrategy implements DataPermissionControlStrategy {
    @Override
    public void process() {
        System.out.println("Admin role can access normal role's data");
    }
}

// 超级管理员角色可以看到所有角色的数据
public class SuperAdminDataPermissionControlStrategy implements DataPermissionControlStrategy {
    @Override
    public void process() {
        System.out.println("Super admin role can access all data");
    }
}
  • 定义用户角色抽象类
public abstract class UserRole {
    protected DataPermissionControlStrategy dataPermissionControlStrategy;

    public void setDataPermissionControlStrategy(DataPermissionControlStrategy dataPermissionControlStrategy) {
        this.dataPermissionControlStrategy = dataPermissionControlStrategy;
    }

    public abstract void viewData();
}
  • 定义具体的用户角色实现类
// 普通角色
public class NormalRole extends UserRole {
    @Override
    public void viewData() {
        System.out.print("Normal role: ");
        dataPermissionControlStrategy.process();
    }
}

// 管理员角色
public class AdminRole extends UserRole {
    @Override
    public void viewData() {
        System.out.print("Admin role: ");
        dataPermissionControlStrategy.process();
    }
}

// 超级管理员角色
public class SuperAdminRole extends UserRole {
    @Override
    public void viewData() {
        System.out.print("Super admin role: ");
        dataPermissionControlStrategy.process();
    }
}
  • 客户端代码演示桥接模式的使用
// 客户端代码演示桥接模式的使用
public class BridgePatternDemo {
    public static void main(String[] args) {
        DataPermissionControlStrategy normalStrategy = new NormalDataPermissionControlStrategy();
        DataPermissionControlStrategy adminStrategy = new AdminDataPermissionControlStrategy();
        DataPermissionControlStrategy superAdminStrategy = new SuperAdminDataPermissionControlStrategy();

        // 创建不同角色的实例
        UserRole normalUser = new NormalRole();
        UserRole adminUser = new AdminRole();
        UserRole superAdminUser = new SuperAdminRole();

        // 设置不同的数据权限控制策略
        normalUser.setDataPermissionControlStrategy(normalStrategy);
        adminUser.setDataPermissionControlStrategy(adminStrategy);
        superAdminUser.setDataPermissionControlStrategy(superAdminStrategy);

        // 视图数据
        normalUser.viewData();  // Output: "Normal role: Normal role can only access own data"
        adminUser.viewData();   // Output: "Admin role: Admin role can access normal role's data"
        superAdminUser.viewData();  // Output: "Super admin role: Super admin role can access all data"
    }
}

四、Java JDBC中使用桥接模式

在Java JDBC中,桥接模式并不是直接使用的设计模式。然而,在JDBC中,可以看到一些类似于桥接模式的结构和思想。

1.DataSource 接口:在JDBC中,DataSource 是一个接口,它提供了获取数据库连接的方法。不同的数据库厂商(比如MySQL、Oracle等)都会有对应的DataSource 实现类。这种设计类似于桥接模式中的抽象部分。

2.Connection 接口和实现类:Connection 接口代表着一个数据库连接,而具体的数据库连接操作是由各个数据库厂商提供的实现类来完成的,比如MySQLConnection、OracleConnection 等。这也体现了桥接模式中的实现部分。

通过接口和实现类的结合,实现了对不同数据库的访问和操作,并且很好地解耦了抽象和实现部分。

// 1.定义厂商DataSource 接口
public interface DataSource {
    Connection getConnection();
}

// 2.MySQLDataSource 实现类
public class MySQLDataSource implements DataSource {
    @Override
    public Connection getConnection() {
        // 返回 MySQL 数据库连接
        return new MySQLConnection();
    }
}

// 3.OracleDataSource 实现类
public class OracleDataSource implements DataSource {
    @Override
    public Connection getConnection() {
        // 返回 Oracle 数据库连接
        return new OracleConnection();
    }
}

// 4.不同的数据库不同的连接方法Connection 接口
public interface Connection {
    void executeQuery(String query);
}

// 5.MySQLConnection 实现类
public class MySQLConnection implements Connection {
    @Override
    public void executeQuery(String query) {
        // 在 MySQL 数据库中执行查询
        System.out.println("Executing query in MySQL database: " + query);
    }
}

// 6.OracleConnection 实现类
public class OracleConnection implements Connection {
    @Override
    public void executeQuery(String query) {
        // 在 Oracle 数据库中执行查询
        System.out.println("Executing query in Oracle database: " + query);
    }
}


http://lihuaxi.xjx100.cn/news/2088272.html

相关文章

TortoiseSVN 报错:The server unexpectedly closed the connetion

前言 CentOS7Linux 安装subversionmod_dav_svn,搭建subversion(svn)服务器 The server unexpectedly closed the connetion 解决办法 重启Apache服务 shell> systemctl restart httpd

每天学习一个Linux命令之tail

每天学习一个Linux命令之tail 在Linux系统中,有许多实用的命令可以帮助我们更高效地管理和操作文件。其中一个非常有用的命令是tail,用于查看文件的末尾内容。本篇博客将详细介绍tail命令的各种选项及其用法。 1. 命令概述 tail命令用于输出指定文件的…

基于深度学习的番茄叶片病害检测系统(含UI界面、yolov8、Python代码、数据集)

项目介绍 项目中所用到的算法模型和数据集等信息如下: 算法模型:     yolov8 yolov8主要包含以下几种创新:         1. 可以任意更换主干结构,支持几百种网络主干。 数据集:     网上下载的数据集&#x…

4、Generator、class类、继承、Set、Map、Promise

一、生成器函数Generator 1、声明generator函数 function* 函数名() { }调用生成器函数 需要next()返回一个generator对象,对象的原型上有一个next(),调用返回对象{value:yield后面的值,done} function* fn() {console.log("我是生成器函数") } let it…

KY276 Problem C

学会了处理1e9素数的新思路&#xff0c;但目前只供挑选最大质因子用 牛客刷题完结撒花&#xff01; 添加链接描述 #include<bits/stdc.h>using namespace std;#define int long long int n; string str;int zhi(int x){int ans 0;for(int i 2; i * i < x; i ){whi…

wsl-oraclelinux 安装 cuda

wsl-oraclelinux 安装 cuda 1. 安装 cuda2. 安装 cuDNN3. 配置环境变量 1. 安装 cuda wget https://developer.download.nvidia.com/compute/cuda/12.4.0/local_installers/cuda_12.4.0_550.54.14_linux.run sudo sh cuda_12.4.0_550.54.14_linux.run2. 安装 cuDNN sudo dnf …

力扣L6--- 两数之和(java版)--2024年3月12日

1.题目 2.知识点 注1&#xff1a;在Java中&#xff0c;数组的长度不是通过调用一个方法获得的&#xff0c;而是通过一个属性直接获得的。因此&#xff0c;正确的语法是nums.length而不是nums.length()。 所以应该使用int m nums.length; 注2&#xff1a;return new int[]{i,…

【力扣白嫖日记】1321.餐馆营业额变化增长

前言 练习sql语句&#xff0c;所有题目来自于力扣&#xff08;https://leetcode.cn/problemset/database/&#xff09;的免费数据库练习题。 今日题目&#xff1a; 1321.餐馆营业额变化增长 表&#xff1a;Customer 列名类型cutomer_idintnamevarcharvisited_ondateamounti…