八、桥接模式

news2024/9/21 12:21:33

文章目录

  • 1 基本介绍
  • 2 案例
    • 2.1 OperatingSystem 抽象类
    • 2.2 LinuxOS 类
    • 2.3 WindowsOS 类
    • 2.4 FileOperation 类
    • 2.5 FileAppender 类
    • 2.6 FileReplicator 类
    • 2.7 Client 类
    • 2.8 Client 类运行结果
    • 2.9 总结
  • 3 各角色之间的关系
    • 3.1 角色
      • 3.1.1 Implementor ( 实现者 )
      • 3.1.2 ConcreteImplementor ( 具体实现者 )
      • 3.1.3 Abstraction ( 抽象化 )
      • 3.1.4 RefinedAbstraction ( 改善后的抽象化 )
    • 3.2 类图
  • 4 注意事项
  • 5 其思想在源码中的使用
  • 6 优缺点
  • 7 适用场景
  • 8 总结


1 基本介绍

桥接模式(Bridge Pattern)是一种 结构型 设计模式,它的主要目的是将 功能实现 分离,使它们都可以 独立地变化

2 案例

本案例的目的是:在 Linux 和 Windows 系统中,对文件使用不同的操作。

两个系统的文件系统都使用 Map 来实现,为了凸显出两个系统的不同,使用不同的 Map 实现类:

  • 在 Linux 系统中使用 ConcurrentHashMap(线程安全)作为 Map 的实现类
  • 在 Windows 系统中使用 HashMap(线程不安全)作为 Map 的实现类。

文件的操作有两种:

  • 基础操作:打开文件、将内容覆盖文件、关闭文件。
  • 进阶操作:需要 复合 基础操作的操作,共有以下三种:
    • 追加操作:将指定内容追加到文件尾部。
    • 复制操作:将指定源文件复制到指定目标文件。
    • 展示操作:展示文件的内容。注意:由于不想导致类的关系太过复杂,所以将其放到基础操作中实现。

2.1 OperatingSystem 抽象类

public abstract class OperatingSystem { // 操作系统 抽象类
    /**
     * 打开文件,如果没有,就新建一个文件
     * @param fileName 待打开文件名
     * @return 文件中的内容
     */
    public abstract String openFile(String fileName);

    /**
     * 修改文件
     * @param fileName 待修改文件名
     * @param content 新的内容
     */
    public abstract void modifyFile(String fileName, String content);

    /**
     * 关闭文件
     * @param fileName 待关闭文件名
     */
    public abstract void closeFile(String fileName);
}

2.2 LinuxOS 类

public class LinuxOS extends OperatingSystem { // Linux 操作系统,它的文件系统线程安全
    /**
     * 文件系统,key 为 文件名,value 为 文件内容
     * 使用 ConcurrentHashMap 保证线程安全
     */
    private final Map<String, String> fileSystem = new ConcurrentHashMap<>();

    @Override
    public String openFile(String fileName) {
        if (!fileSystem.containsKey(fileName)) { // 如果不存在某个文件
            fileSystem.put(fileName, ""); // 则创建一个空文件
        }
        return fileSystem.get(fileName);
    }

    @Override
    public void modifyFile(String fileName, String content) {
        fileSystem.put(fileName, content);
    }

    @Override
    public void closeFile(String fileName) {
        // 在本案例中无需实现
    }
}

2.3 WindowsOS 类

public class WindowsOS extends OperatingSystem { // Windows 操作系统,它的文件系统线程不安全
    /**
     * 文件系统,key 为 文件名,value 为 文件内容
     * 使用 HashMap 可能会出现线程不安全的问题
     */
    private final Map<String, String> fileSystem = new HashMap<>();

    @Override
    public String openFile(String fileName) {
        if (!fileSystem.containsKey(fileName)) { // 如果不存在某个文件
            fileSystem.put(fileName, ""); // 则创建一个空文件
        }
        return fileSystem.get(fileName);
    }

    @Override
    public void modifyFile(String fileName, String content) {
        fileSystem.put(fileName, content);
    }

    @Override
    public void closeFile(String fileName) {
        // 在本案例中无需实现
    }
}

2.4 FileOperation 类

public class FileOperation { // 基础文件操作的类
    private final OperatingSystem os; // 操作系统

    public FileOperation(OperatingSystem os) {
        this.os = os;
    }

    /**
     * 打开文件,如果没有,就新建一个文件
     * @param fileName 待打开文件名
     * @return 文件中的内容
     */
    public String openFile(String fileName) {
        return os.openFile(fileName);
    }

    /**
     * 修改文件
     * @param fileName 待修改文件名
     * @param content 新的内容
     */
    public void modifyFile(String fileName, String content) {
        os.modifyFile(fileName, content);
    }

    /**
     * 关闭文件
     * @param fileName 待关闭文件名
     */
    public void closeFile(String fileName) {
        os.closeFile(fileName);
    }

    /**
     * 显示文件的内容
     * @param fileName 待显示内容的文件名
     */
    public void displayFile(String fileName) {
        String content = os.openFile(fileName);
        System.out.println("-----------------------------------------------------------");
        System.out.println("[" + fileName + "]");
        System.out.println(content);
        System.out.println("-----------------------------------------------------------");
        os.closeFile(fileName);
    }
}

2.5 FileAppender 类

public class FileAppender extends FileOperation { // 处理 追加内容操作 的类
    public FileAppender(OperatingSystem os) {
        super(os);
    }

    /**
     * 将指定内容追加到文件尾部
     * @param fileName 待追加文件名
     * @param appendContent 追加内容
     */
    public void append(String fileName, String appendContent) {
        String content = super.openFile(fileName); // 指定文件原有的内容
        content += appendContent; // 将新内容追加到原有的内容后
        super.modifyFile(fileName, content); // 修改文件
        super.closeFile(fileName); // 关闭文件
    }
}

2.6 FileReplicator 类

public class FileReplicator extends FileOperation { // 处理 复制文件操作 的类
    public FileReplicator(OperatingSystem os) {
        super(os);
    }

    /**
     * 将 源文件的内容 拷贝到 目标文件中
     * @param srcFileName 源文件名称
     * @param dstFileName 目标文件名称
     */
    public void replicate(String srcFileName, String dstFileName) {
        String srcContent = super.openFile(srcFileName); // 打开源文件,并获取源文件的内容
        super.openFile(dstFileName); // 打开目标文件
        super.modifyFile(dstFileName, srcContent); // 将 目标文件的内容 修改为 源文件的内容
        super.closeFile(dstFileName); // 关闭目标文件
        super.closeFile(srcFileName); // 关闭源文件
    }
}

2.7 Client 类

public class Client { // 使用 两种系统 分别执行 两种操作 的客户端
    public static void main(String[] args) {
    	// 文件名称
        final String srcFileName = "src.txt"; // 源文件
        final String dstFileName = "dst.txt"; // 目标文件

		// 操作系统
        OperatingSystem linuxOS = new LinuxOS();
        
        OperatingSystem windowsOS = new WindowsOS();

		// 各种文件操作
        FileOperation linuxFileOperation = new FileOperation(linuxOS);
        FileAppender linuxFileAppender = new FileAppender(linuxOS);
        FileReplicator linuxFileReplicator = new FileReplicator(linuxOS);
        
        FileOperation windowsFileOperation = new FileOperation(windowsOS);
        FileAppender windowsFileAppender = new FileAppender(windowsOS);
        FileReplicator windowsFileReplicator = new FileReplicator(windowsOS);

        // 先分别在 Linux 和 Windows 系统中创建一个文件,并写入不同内容,之后查看内容
        System.out.println("操作一:创建文件并写入内容");

        linuxFileOperation.openFile(srcFileName);
        linuxFileOperation.modifyFile(srcFileName, "Hello, LinuxOS!");
        linuxFileOperation.closeFile(srcFileName);
        linuxFileOperation.displayFile(srcFileName);

        windowsFileOperation.openFile(srcFileName);
        windowsFileOperation.modifyFile(srcFileName, "Hello, WindowsOS!");
        windowsFileOperation.closeFile(srcFileName);
        windowsFileOperation.displayFile(srcFileName);

        System.out.println("===============================" +
                "=============================="); // 分隔

        // 然后给这两个文件追加不同的内容,并查看
        System.out.println("操作二:给文件追加内容");

        linuxFileAppender.append(srcFileName,
                "\nI am a programmer using Linux system.");
        linuxFileOperation.displayFile(srcFileName);

        windowsFileAppender.append(srcFileName,
                "\nI am a programmer using Windows system.");
        windowsFileOperation.displayFile(srcFileName);

        System.out.println("===============================" +
                "=============================="); // 分隔

        // 最后将两个文件复制到新的文件中,并查看
        System.out.println("操作三:复制文件内容");

        linuxFileReplicator.replicate(srcFileName, dstFileName);
        linuxFileOperation.displayFile(dstFileName);

        windowsFileReplicator.replicate(srcFileName, dstFileName);
        windowsFileOperation.displayFile(dstFileName);
    }
}

2.8 Client 类运行结果

操作一:创建文件并写入内容
-----------------------------------------------------------
[src.txt]
Hello, LinuxOS!
-----------------------------------------------------------
-----------------------------------------------------------
[src.txt]
Hello, WindowsOS!
-----------------------------------------------------------
=============================================================
操作二:给文件追加内容
-----------------------------------------------------------
[src.txt]
Hello, LinuxOS!
I am a programmer using Linux system.
-----------------------------------------------------------
-----------------------------------------------------------
[src.txt]
Hello, WindowsOS!
I am a programmer using Windows system.
-----------------------------------------------------------
=============================================================
操作三:复制文件内容
-----------------------------------------------------------
[dst.txt]
Hello, LinuxOS!
I am a programmer using Linux system.
-----------------------------------------------------------
-----------------------------------------------------------
[dst.txt]
Hello, WindowsOS!
I am a programmer using Windows system.
-----------------------------------------------------------

2.9 总结

在本案例中,OperatingSystem 抽象类 和 LinuxOS, WindowsOS 类组成了 实现FileOperation, FileAppender, FileReplicator 类组成了 功能,使用 FileOperation 类将 实现功能 分隔开来。

由于不想再增加本案例的复杂度,所以将 展示操作 放到本类中,依照本模式的思想,应该将 展示操作 也放到一个单独的类中,大家可以将 写一个类来处理 展示操作 当作练习。

在添加新的功能(或实现)时,无需为其添加对应的实现(或功能)类

  • 如果想要添加一个新功能, 需要继承 功能 体系中的某个类即可。如将 展示操作 写到独立的类中。
  • 如果想要添加一个新实现, 需要继承 OperatingSystem 抽象类即可。如添加 MacOS 类。

3 各角色之间的关系

3.1 角色

3.1.1 Implementor ( 实现者 )

该角色负责 定义 用于实现 Abstraction 角色的接口的 方法,位于 实现 体系的顶端。在本案例中,OperatingSystem 抽象类扮演该角色。

3.1.2 ConcreteImplementor ( 具体实现者 )

该角色负责 实现 在 Implementor 角色中定义的 方法。在本案例中,LinuxOS, WindowsOS 类都扮演了该角色。

3.1.3 Abstraction ( 抽象化 )

该角色负责 使用 Implementor 角色的方法,定义基本的功能,位于 功能 体系的顶端。此外,还需要保存 Implementor 角色的实例。在本案例中,FileOperation 类扮演了该角色。

3.1.4 RefinedAbstraction ( 改善后的抽象化 )

该角色负责 在 Abstraction 角色的基础上添加新功能。在本案例中,FileAppender, FileReplicator 类都扮演了该角色。

3.2 类图

alt text
说明:

  • Implementor 角色中的方法 method1(), method2()abstract 的,强制其子类实现。
  • Implementor 角色也可以是接口,这样一来,ConcreteImplementor 角色与它之间的关系就是 实现 了。
  • Abstraction 角色 和 Implementor 角色 中间的 聚合 关系 就代表了 桥接模式 的 桥接

4 注意事项

  • 分离 功能 和 实现:在使用桥接模式进行设计时,需要 辨析 系统的 功能实现,从而将其分离开来。这无疑增加了设计难度。
  • 使用范围局限性:如果想要使用桥接模式,则系统中必须存在 两个或多个独立变化的维度,并且这些维度需要 独立扩展。如果系统中没有这样的应用场景,强行使用桥接模式可能会导致设计 过度复杂,反而降低系统的可维护性和可扩展性。

5 其思想在源码中的使用

JDBC(Java Database Connectivity)在使用桥接模式时,主要体现在它如何处理 数据库驱动的加载数据库的使用 上:

  • 实现体系
    • 抽象实现:在 JDBC 中,java.sql.Driver 接口扮演了 Implementor 角色。它定义了一个驱动应该实现的接口,包括如何连接到数据库等方法。
    • 具体实现:各个数据库厂商提供的驱动类(如 com.mysql.cj.jdbc.Driver)(直接或间接地) 实现了 java.sql.Driver 接口,扮演 ConcreteImplementor 角色。这些驱动类包含了与特定数据库交互的具体逻辑。
  • 功能体系——DriverManager 类
    • DriverManager 类在 JDBC 中起到了 桥梁 的作用,它维护了一个已注册的 JDBC 驱动列表(registeredDrivers)。当通过 Class.forName() 方法加载数据库驱动时,实际上是在调用 该驱动类的静态代码块,其中包含了 将自身实例注册到 DriverManagerregisteredDrivers 列表中 的代码。
    • 当调用 DriverManager.getConnection() 方法获取数据库连接时,DriverManager 会遍历 registeredDrivers 列表,尝试使用列表中的每个驱动与数据库建立连接,直到找到成功的连接或所有驱动都尝试失败为止。
    • 此外,DriverManager 类独立承担了所有的 功能,没有将更复杂的功能交给子类实现,这就是 JDBC 使用的桥接模式 与 常规桥接模式 的不同之处。
  • 实现 与 功能 的分离
    • 实现——数据库驱动:不同的数据库需要不同的驱动来实现与 JDBC 的交互。
    • 功能——数据库操作DriverManager 类提供了所有的数据库操作。
  • 桥接模式的优势
    • 扩展性强:由于功能与实现分离,当需要支持新的数据库时,只需提供新的驱动实现即可,无需修改现有的 JDBC 代码。
    • 耦合度低:JDBC 的抽象接口与具体数据库实现之间通过 DriverManager 进行桥接,降低了它们之间的耦合度。

6 优缺点

优点

  • 提高可扩展性:桥接模式将 功能实现 分离,使得它们可以 独立地变化。这意味着我们可以在不修改抽象代码的情况下,通过增加新的实现类来扩展系统的功能。
  • 符合开闭原则:桥接模式符合开闭原则,即对扩展开放,对修改关闭。这意味着我们可以在不修改现有代码的情况下,通过扩展系统来满足新的需求。
  • 降低系统的耦合度:通过桥接模式,我们可以减少类之间的直接依赖关系,降低系统的耦合度。这有助于提高系统的 可维护性可测试性
  • 支持多种变化:当一个类存在 多个角度的变化 时(例如,一个类既需要支持不同的操作,又需要支持不同的实现方式),桥接模式可以很好地处理这种情况,因为它允许在多个维度上进行变化。

缺点

  • 增加系统的复杂性:桥接模式增加了系统的抽象层次,使得系统的理解和实现变得更加复杂。特别是当系统中存在大量的抽象类和实现类时,这种复杂性可能会变得难以管理。
  • 增加设计的难度:在设计初期,正确地识别出哪些部分(功能部分)应该 被抽象化,哪些部分(实现部分)应该 被实现化并不是一件容易的事情。这需要设计者对系统有深入的理解和丰富的经验。
  • 增加对象的数量:由于桥接模式将 功能 和 实现 分离,因此在运行时可能会产生大量的对象。这可能会增加系统的内存开销,并降低系统的性能。
  • 过度设计:在某些情况下,可能会存在过度使用桥接模式的风险。如果系统中的变化维度并不明显,或者变化的可能性很小,那么使用桥接模式可能会导致过度设计,增加不必要的复杂性。

7 适用场景

  • 数据库连接 与 驱动:在连接数据库时,桥接模式可以用于分离 数据库连接接口(功能)和 不同的数据库驱动程序(实现)。这样,在不修改连接代码的情况下,可以轻松地切换不同的数据库。例如,在 JDBC(Java Database Connectivity)中,Driver 接口作为抽象部分,不同的数据库厂商(如 MySQL、Oracle)提供的驱动实现类作为实现部分,通过桥接模式实现数据库的灵活切换。
  • 日志记录 与 输出目标:在记录日志时,桥接模式可以将 日志记录器(功能)与 不同的日志输出目标(如控制台、文件、数据库等,实现)解耦。这样,可以在不修改日志记录代码的情况下,轻松地切换日志输出目标。例如,在日志框架中,可以定义一个日志记录器接口,不同的日志实现类(如控制台日志实现类、文件日志实现类等)分别实现该接口,通过桥接模式实现日志记录的灵活配置。
  • 消息队列 与 协议:在使用消息队列时,桥接模式可以用于分离 消息队列客户端接口(功能)和 不同的消息队列协议(如 RocketMQ、Kafka等,实现)。这样,可以在不修改消息队列代码的情况下,轻松地切换不同的消息队列协议。例如,在消息队列客户端框架中,可以定义一个消息队列客户端接口,不同的消息队列协议实现类分别实现该接口,通过桥接模式实现消息队列的灵活选择。
  • 图形用户界面(GUI):在图形用户界面(GUI)框架中,桥接模式可以用于分离 GUI 组件的抽象表示具体渲染实现。例如,在 Swing 或 JavaFX 等 GUI 框架中,可以将组件的抽象行为(如按钮点击事件)与具体的渲染逻辑(如按钮的外观和样式)分离,通过桥接模式实现 GUI 组件的灵活定制和扩展。
  • 插件系统:在插件系统中,桥接模式可以用于分离 插件的接口定义具体的插件实现。这样,可以在不修改主程序代码的情况下,通过添加新的插件实现来扩展程序的功能。插件系统广泛应用于各种软件开发中,如 IDE(集成开发环境)、游戏平台等,桥接模式为这些系统提供了良好的扩展性和灵活性。

8 总结

桥接模式 是一种 结构型 设计模式,它的主要目的是将 功能实现 分离,使它们都可以 独立地变化。这种分离方式增强了系统的 灵活性可扩展性,允许在不修改抽象代码的情况下增加新的实现。桥接模式通过 聚合关系 而非 继承关系 来实现这一点,从而减少了类之间的耦合。

此外,在使用本模式之前一定要理清系统的逻辑,认清哪部分是 功能,哪部分是 实现。如果功能和实现变化的程度不大,则 不要为了使用桥接模式而使用桥接模式,这时简单一点,将其耦合起来比较方便。

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

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

相关文章

微信答题小程序产品研发-UI界面设计

高保真原型虽然已经很接近产品形态了&#xff0c;但毕竟还不能够直接交付给开发。这时就需要UI设计师依据之前的原型设计&#xff0c;进一步细化和实现界面的视觉元素&#xff0c;包括整体视觉风格、颜色、字体、图标、按钮以及交互细节优化等。 UI设计不仅关系到用户的直观感…

1.c#(winform)编程环境安装

目录 安装vs创建应用帮助查看器安装与使用&#xff08; msdn&#xff09; 安装vs 安装什么版本看个人心情&#xff0c;或者公司开发需求需要 而本栏全程使用vs2022进行开发c#&#xff0c;着重讲解winform桌面应用开发 使用***.net framework***开发 那先去官网安装企业版的vs…

这一文,关于Java泛型的点点滴滴 一

作为一个 Java 程序员&#xff0c;用到泛型最多的&#xff0c;我估计应该就是这一行代码&#xff1a; List<String> list new ArrayList<>();这也是所有 Java 程序员的泛型之路开始的地方啊。 不过本文讲泛型&#xff0c;先不从这里开始讲&#xff0c;而是再往前…

CVPR 2024最佳论文分享:Mip-Splatting: 无混叠3D高斯溅射

本推文详细介绍了CVPR 2024最佳论文提名《Mip-Splatting: Alias-free 3D Gaussian Splatting》。该论文的第一作者为 Zehao Yu&#xff08;图宾根大学在读博士&#xff0c;导师&#xff1a;Andreas Geiger &#xff09;。论文提出了一种名为Mip-Splatting的方法&#xff0c;用于…

树和二叉树(不用看课程)

1. 树 1.1 树的概念与结构 树是⼀种非线性的数据结构&#xff0c;它是由 n&#xff08;n>0&#xff09; 个有限结点组成⼀个具有层次关系的集合。把它叫做树是因为它看起来像⼀棵倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;而叶朝下的。 • 有⼀个特殊的结点&am…

大语言模型赋能设施农业:透过“智慧大脑“看智能环境调控

&#xff08;文/ 于景鑫 北京市农林科学院&#xff09;在上一篇专栏文章中,我们从宏观视角探讨了大语言模型为设施农业插上腾飞之翼的广阔前景。而要真正实现这一愿景,还需要在微观层面深入剖析LLM的技术原理和应用路径。本文将聚焦设施农业的核心环节之一——环境调控,看看&qu…

【解决】ubuntu20.04 root用户无法SSH登陆问题

Ubuntu root用户无法登录的问题通常可以通过修改‌SSH配置文件和系统登录配置来解决。 修改SSH配置文件 sudo vim /etc/ssh/sshd_config 找到 PermitRootLogin 设置&#xff0c;并将其值更改为 yes 以允许root用户通过SSH登录 保存并关闭文件之后&#xff0c;需要重启SSH服务…

Xshell、XFTP的安装配置及其使用

Xshell、XFTP的安装配置及其使用 Xshell的优点 安全远程连接&#xff1a; Xshell 使用 SSH 协议等安全协议进行远程连接&#xff0c;确保数据传输的加密和安全性。多会话管理&#xff1a; 用户可以同时管理多个远程连接&#xff0c;方便在不同服务器之间切换和操作。终端仿真…

html+css+js 实现马赛克背景按钮

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享htmlcss 绚丽效果&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 文…

python爬虫-事件触发机制

今天想爬取一些政策&#xff0c;从政策服务 (smejs.cn) 这个网址爬取&#xff0c;html源码找不到链接地址&#xff0c;通过浏览器的开发者工具&#xff0c;点击以下红框 分析预览可知想要的链接地址的id有了&#xff0c;进行地址拼接就行 点击标头可以看到请求后端服务器的api地…

女人内裤怎么洗才是最干净?内衣裤洗衣机怎么样?哪个牌子更好?

最近刚好用到一款比较好用的洗内衣裤洗衣机&#xff01;如果你也和我一样有洗内衣裤烦恼的&#xff0c;或者可以看看&#xff01; 内衣裤作为贴身穿的衣服&#xff0c;我是不会把它和外衣一起清洗的&#xff0c;而家里面的大洗衣机已经担起了清洗外衣的工作&#xff01; 朋友们…

React Router-v6.25.1

以下例子是根据vitereactts构建的&#xff0c;使用路由前先安装好这些环境&#xff01;&#xff01;&#xff01;&#xff01; 1、路由的简单使用 首先要创建一个浏览器路由器并配置我们的第一个路由。这将为我们的 Web 应用启用客户端路由。 该main.jsx文件是入口点。打开它…

【杰理蓝牙开发】AC695x 音频部分

本文主要记录 杰理蓝牙audio接口的使用&#xff0c;包括ADC和DAC原理的介绍和API接口的使用。 【杰理蓝牙开发】AC695x 音频部分 0. 个人简介 && 授权须知1. ADC【音频数据采集】硬件部分1.1 单片机引脚1.2 硬件电路设计1.3 MIC 输入通路解释 2. 【DAC】音频信号编解码…

GLSL教程 第9章:计算着色器

目录 9.1 计算着色器的基本概念 计算着色器的主要特点&#xff1a; 9.2 计算着色器的基础知识 1. 创建计算着色器 计算着色器代码&#xff1a; 2. 编译和链接计算着色器 示例代码&#xff1a; 3. 执行计算着色器 示例代码&#xff1a; 9.3 实现并行计算和数据并行处理…

51单片机-第五节-串口通信

1.什么是串口&#xff1f; 串口是通讯接口&#xff0c;实现两个设备的互相通信。 单片机自带UART&#xff0c;其中引脚有TXD发送端&#xff0c;RXD接收端。且电平标准为TTL&#xff08;5V为1,0V为0&#xff09;。 2.常见电平标准&#xff1a; &#xff08;1&#xff09;TTL电…

Mysql中如何实现两列的值互换?给你提供些思路。

文章目录 Mysql中如何实现两列的值互换1、第一感觉此sql应该能处理问题了2、需要一个地方存要替换的值&#xff0c;不然两列搞不定。2.1 加第三列&#xff1f;&#xff08;能解决&#xff0c;但是看起来呆呆&#xff09;2.2 上临时表&#xff08;搞点弯路走走&#xff09; 示例…

C语言画蜡烛图

GPT-4o (OpenAI) 在C语言中&#xff0c;绘制蜡烛图&#xff08;Candlestick Chart&#xff09;不是直接的任务&#xff0c;因为C语言本身不包含高级图形绘制库。然而&#xff0c;可以通过某些图形库来完成这项任务&#xff0c;例如使用GTK、SDL、OpenGL等。 以下是通过GTK库绘…

【iOS】—— retain\release实现原理和属性关键字

【iOS】—— retain\release实现原理和属性关键字 1. retain\reelase实现原理1.1 retain实现原理1.2 release实现原理 2. 属性关键字2.1 属性关键字的分类2.2 内存管理关键字2.2.1 weak2.2.2 assgin2.3.3 strong和copy 2.4 线程安全的关键字2.5 修饰变量的关键字2.5.1常量const…

北京率先建设AI原生城市,力争明年推出百个优秀行业大模型产品

7月26日&#xff0c;《北京市推动“人工智能”行动计划&#xff08;2024-2025年&#xff09;》&#xff08;简称《行动计划》&#xff09;正式向社会发布&#xff0c;新京报记者在北京市发展和改革委员会举行的新闻发布会上获悉&#xff0c;北京将率先建设AI原生城市&#xff0…