【设计模式——学习笔记】23种设计模式——桥接模式Bridge(原理讲解+应用场景介绍+案例介绍+Java代码实现)

问题引入

现在对不同手机类型的不同品牌实现操作编程(比如:开机、关机、上网,打电话等),如图

在这里插入图片描述

【对应类图】

在这里插入图片描述

【分析】

  • 扩展性问题(类爆炸),如果我们再增加手机的样式(旋转式),就需要增加各个品牌手机的类,同样如果我们增加一个手机品牌,也要在各个手机样式类下增加。
  • 违反了单一职责原则,当我们增加手机样式时,要同时增加所有品牌的手机,这样增加了代码维护成本
  • 解决方案:使用桥接模式

介绍

基础介绍

  • 桥接模式(Bridge模式)是指: 将实现与抽象放在两个不同的类层次中,使两个层次可以独立改变。
  • Bridge 的意思是“桥梁”。就像在现实世界中,桥梁的功能是将河流的两侧连接起来一样Bridge 模式的作用也是将两样东西连接起来,它们分别是类的功能层次结构和类的实现层次结构。
  • 是一种结构型设计模式
  • Bridge模式基于类的最小设计原则(实现功能的同时,让类尽可能少),通过使用封装、聚合及继承等行为让不同的类承担不同的职责。它的主要特点是把抽象(Abstraction)与行为实现(lmplementation)分离开来,从而可以保持各部分的独立性以及应对他们的功能扩展

类的功能层次结构

假设现在有一个类Something。当我们想在Something中增加新功能时(想增加一个具体方法时),会编写一个Something类的子类(派生类),如SomethingGood类。这样就构成了一个小小的类层次结构。

在这里插入图片描述

父类本身具备一些基本功能,子类在继承父类的功能之外,还可以添加新的功能。这种层次结构被称为“类的功能层次结构”。如果需要在SomethingGood的基础上继续增加新的功能,再写SomethingGood的子类即可。

在这里插入图片描述

注意:类的层次结构关系不应当过深

类的实现层次结构

抽象类声明了一些抽象方法,定义了接口(API),然后子类负责去实现这些抽象方法。父类的任务是通过声明抽象方法的方式定义接口(API),而子类的任务是实现抽象方法。正是由于父类和子类的这种任务分担,我们才可以编写出具有高可替换性的类。

当子类Concreteclass实现了父类Abstractclass类的抽象方法时,它们之间就构成了一个小小的层次结构。

在这里插入图片描述

这种类的层次结构(类的实现层次结构)并非用于增加功能,并不方便我们增加新的方法。它的真正作用是帮助我们实现下面这样的任务分担

  • 类通过声明抽象方法来定义接口(API)
  • 子类通过实现具体方法来实现接口(API)

一个抽象类可以有多种子实现类

在这里插入图片描述

层次结构分离

当类的层次结构只有一层时,功能层次结构与实现层次结构是混杂在一个层次结构中的。这样很容易使类的层次结构变得复杂,也难以透彻地理解类的层次结构。因为自己难以确定究竟应该在类的哪一个层次结构中去增加子类。

因此,我们需要将“类的功能层次结构”与“类的实现层次结构”分离为两个独立的类层次结构。当然,如果只是简单地将它们分开,两者之间必然会缺少联系。所以我们还需要在它们之间搭建一座桥梁,这就是桥接模式。

类图

在这里插入图片描述

抽象类和接口是聚合关系,也是调用和被调用关系

应用场景

对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用,如以下场景:

  • JDBC驱动程序
  • 银行转账系统
    • 转账分类: 网上转账,柜台转账,AMT转账
    • 转账用户类型: 普通用户,银卡用户,金卡用户
  • 消息管理
    • 消息类型: 即时消息,延时消息
    • 消息分类: 手机短信,邮件消息,QQ消息…

案例

案例一

类图

在这里插入图片描述

代码实现

【接口】

package com.atguigu.bridge;

/**
 * 接口
 */
public interface Brand {
   void open();
   void close();
   void call();
}

【接口实现类:小米手机】

package com.atguigu.bridge;

public class XiaoMi implements Brand {

   @Override
   public void open() {
      System.out.println(" 小米手机开机 ");
   }

   @Override
   public void close() {
      System.out.println(" 小米手机关机 ");
   }

   @Override
   public void call() {
      System.out.println(" 小米手机打电话 ");
   }

}

【接口实现类:Vivo手机】

package com.atguigu.bridge;

public class Vivo implements Brand {

   @Override
   public void open() {
      System.out.println(" Vivo手机开机 ");
   }

   @Override
   public void close() {
      System.out.println(" Vivo手机关机 ");
   }

   @Override
   public void call() {
      System.out.println(" Vivo手机打电话 ");
   }

}

【抽象类】

package com.atguigu.bridge;

public abstract class Phone {

   /**
    * 聚合品牌
    */
   private Brand brand;

   /**
    * 构造器
    * @param brand
    */
   public Phone(Brand brand) {
      super();
      this.brand = brand;
   }

   protected void open() {
      this.brand.open();
   }
   protected void close() {
      this.brand.close();
   }
   protected void call() {
      this.brand.call();
   }

}

【抽象类子类:折叠手机】

package com.atguigu.bridge;


/**
 * 折叠式手机类,继承 抽象类 Phone
 */
public class FoldPhone extends Phone {

   //构造器
   public FoldPhone(Brand brand) {
      super(brand);
   }

   public void open() {
      System.out.println(" 折叠样式手机 ");
      // 实际上调用的是具体品牌(如Xiaomi)的开机方法,抽象类Phone充当桥接作用
      super.open();
   }

   public void close() {
      System.out.println(" 折叠样式手机 ");
      super.close();
   }

   public void call() {
      System.out.println(" 折叠样式手机 ");
      super.call();
   }
}

【抽象类子类:直立手机】

package com.atguigu.bridge;

public class UpRightPhone extends Phone {

   //构造器
   public UpRightPhone(Brand brand) {
      super(brand);
   }

   public void open() {
      System.out.println(" 直立样式手机 ");
      super.open();
   }

   public void close() {
      System.out.println(" 直立样式手机 ");
      super.close();
   }

   public void call() {
      System.out.println(" 直立样式手机 ");
      super.call();
   }
}

【客户端】

package com.atguigu.bridge;

public class Client {

   public static void main(String[] args) {

      //获取折叠式手机 (样式 + 品牌 才是具体的手机)
      Phone phone1 = new FoldPhone(new XiaoMi());
      phone1.open();
      phone1.call();
      phone1.close();

      System.out.println("==============");

      Phone phone2 = new FoldPhone(new Vivo());
      phone2.open();
      phone2.call();
      phone2.close();

      System.out.println("==============");

      UpRightPhone phone3 = new UpRightPhone(new XiaoMi());
      phone3.open();
      phone3.call();
      phone3.close();

      System.out.println("==============");

      UpRightPhone phone4 = new UpRightPhone(new Vivo());
      phone4.open();
      phone4.call();
      phone4.close();
   }

}

【运行】

折叠样式手机 
 小米手机开机 
 折叠样式手机 
 小米手机打电话 
 折叠样式手机 
 小米手机关机 
==============
 折叠样式手机 
 Vivo手机开机 
 折叠样式手机 
 Vivo手机打电话 
 折叠样式手机 
 Vivo手机关机 
==============
 直立样式手机 
 小米手机开机 
 直立样式手机 
 小米手机打电话 
 直立样式手机 
 小米手机关机 
==============
 直立样式手机 
 Vivo手机开机 
 直立样式手机 
 Vivo手机打电话 
 直立样式手机 
 Vivo手机关机 

Process finished with exit code 0

分析

  • 无论增加一个样式或者增加一个新的手机品牌,都只需要增加一个类

案例二

类图

在这里插入图片描述

Display使用DispalyImpl的方法来完成功能,StringDisplayImpl负责DispalyImpl方法的具体实现,CountDisplay用来完成更多的功能

代码实现

【类的功能层次结构:Display】

package com.atguigu.bridge.Sample;

/**
 * 类的功能层次最上层
 */
public class Display {
    /**
     * 实现了Display类的具体功能的实例 (桥梁)
     */
    private DisplayImpl impl;
    public Display(DisplayImpl impl) {
        this.impl = impl;
    }

    /**
     * 显示前的处理
     */
    public void open() {
        impl.rawOpen();
    }

    /**
     * 显示处理
     */
    public void print() {
        impl.rawPrint();
    }

    /**
     * 显示后的处理
     */
    public void close() {
        impl.rawClose();
    }

    /**
     * 调用上面的三个方法来进行显示
     */
    public final void display() {
        open();
        print();                    
        close();
    }
}

private DisplayImpl impl;使用了“委托”关系,而不是使用“继承”关系。继承是强关联关系,委托是弱关联关系。使用委托更方便扩展。

【类的功能层次结构:CountDisplay】

package com.atguigu.bridge.Sample;

/**
 * 在Display类的基础上增加新功能
 */
public class CountDisplay extends Display {
    public CountDisplay(DisplayImpl impl) {
        super(impl);
    }

    /**
     * 循环显示times次
     *
     * @param times
     */
    public void multiDisplay(int times) {
        open();
        for (int i = 0; i < times; i++) {
            print();
        }
        close();
    }
}

【类的实现层次结构:上层】

package com.atguigu.bridge.Sample;

/**
 * 类的实现层次最上层
 */
public abstract class DisplayImpl {
    public abstract void rawOpen();
    public abstract void rawPrint();
    public abstract void rawClose();
}

【类的实现层次结构:下层】

package com.atguigu.bridge.Sample;

/**
 * 类的实现层次结构,具体实现类
 */
public class StringDisplayImpl extends DisplayImpl {
    /**
     * 要显示的字符串
     */
    private String string;
    /**
     * 以字节单位计算出的字符串的宽度
     */
    private int width;

    /**
     * 构造方法
     *
     * @param string 构造函数接收要显示的字符串string
     */
    public StringDisplayImpl(String string) {
        // 将它保存在字段中
        this.string = string;
        // 把字符串的宽度也保存在字段中,以供使用
        this.width = string.getBytes().length;
    }

    public void rawOpen() {
        printLine();
    }

    public void rawPrint() {
        // 前后加上"|"并显示
        System.out.println("|" + string + "|");
    }

    public void rawClose() {
        printLine();
    }

    private void printLine() {
        // 显示用来表示方框的角的"+"
        System.out.print("+");
        // 显示width个"-"
        for (int i = 0; i < width; i++) {
            // 将其用作方框的边框
            System.out.print("-");                    
        }
        // 显示用来表示方框的角的"+"
        System.out.println("+");
    }
}

【主类】

package com.atguigu.bridge.Sample;

public class Main {
    public static void main(String[] args) {
        Display d1 = new Display(new StringDisplayImpl("Hello, China."));
        Display d2 = new CountDisplay(new StringDisplayImpl("Hello, World."));
        CountDisplay d3 = new CountDisplay(new StringDisplayImpl("Hello, Universe."));
        d1.display();
        System.out.println();
        d2.display();
        System.out.println();
        d3.display();
        d3.multiDisplay(5);
    }
}

【运行】

+-------------+
|Hello, China.|
+-------------+

+-------------+
|Hello, World.|
+-------------+

+----------------+
|Hello, Universe.|
+----------------+
+----------------+
|Hello, Universe.|
|Hello, Universe.|
|Hello, Universe.|
|Hello, Universe.|
|Hello, Universe.|
+----------------+

Process finished with exit code 0

拓展一

在上述示例程序中增加一个类,实现“显示字符串若干(随机)次”的功能。

【新增类】

package com.atguigu.bridge.A1;

import java.util.Random;

public class RandomCountDisplay extends CountDisplay {
    private Random random = new Random();

    public RandomCountDisplay(DisplayImpl impl) {
        super(impl);
    }

    public void randomDisplay(int times) {
        multiDisplay(random.nextInt(times));
    }
}

【主类】

package com.atguigu.bridge.A1;

public class Main {
    public static void main(String[] args) {
        RandomCountDisplay d = new RandomCountDisplay(new StringDisplayImpl("Hello, China."));
        d.randomDisplay(10);
    }
}

扩展二

在上述示例程序中增加一个类,实现“显示文本文件的内容”的功能。

【新增类】

package com.atguigu.bridge.A2;

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;

public class FileDisplayImpl extends DisplayImpl {
    private String filename;
    private BufferedReader reader;
    /**
     * 循环显示的极限(缓存大小限制)
     */
    private final int MAX_READAHEAD_LIMIT = 4096;  
    public FileDisplayImpl(String filename) {
        this.filename = filename;
    }
    public void rawOpen() {
        try {
            reader = new BufferedReader(new FileReader(filename));
            reader.mark(MAX_READAHEAD_LIMIT);
        } catch (IOException e) {
            e.printStackTrace();
        }
        // 装饰框
        System.out.println("=-=-=-=-=-= " + filename + " =-=-=-=-=-="); 
    }
    public void rawPrint() {
        try {
            String line;
            reader.reset(); // 回到mark的位置
            while ((line = reader.readLine()) != null) {
                System.out.println("> " + line);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    public void rawClose() {
        // 装饰框
        System.out.println("=-=-=-=-=-= "); 
        try {
            reader.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

【主类】

package com.atguigu.bridge.A2;

public class Main {
    public static void main(String[] args) {
        CountDisplay d = new CountDisplay(new FileDisplayImpl("star.txt"));
        d.multiDisplay(3);
    }
}

登场角色

  • Abstraction(抽象化):位于类的功能层次结构的最上层。它使用Implementor角色的方法定义了基本的功能。该角色中保存了Implementor 角色的实例
  • RefinedAbstraction(改善后的抽象化):在Abstraction角色的基础上增加了新功能
  • Implementor(实现者):位于“类的实现层次结构”的最上层。它定义了用于实现Abstraction 角色的接口(API)的方法
  • Concretelmplementor( 具体实现者):负责实现在Implementor 角色中定义的接口(API)

在这里插入图片描述

桥接模式在JDBC源码中的应用

在这里插入图片描述

总结

  • 实现了抽象和实现部分的分离,从而极大的提高了系统的灵活性,让抽象部分和实现部分独立开来,这有助于系统进行分层设计,从而产生更好的结构化系统
  • 分离之后代码的扩展性更强,当想要增加功能时,只需要在“类的功能层次结构”一侧增加类即可,不必对“类的实现层次结构”做任何修改。而且,增加后的功能可以被“所有的实现”使用(例如,我们可以将“类的功能层次结构”应用于软件所运行的操作系统上。如果我们将某个程序中依赖于操作系统的部分划分为 Windows版、Macintosh 版、Unix 版,那么我们就可以用 Bridge模式中的“类的实现层次结构”来表现这些依赖于操作系统的部分。也就是说,我们需要编写一个定义这些操作系统的共同接口(API)的Implementor角色,然后编写Windows版、Macintosh版Unix版的3个Concretelmplementor角色。这样一来,无论在“类的功能层次结构”中增加多少个功能,它们都可以工作于这3个操作系统上。)
  • 对于系统的高层部分,只需要知道抽象部分和实现部分的接口就可以了,其它的部分由具体业务来完成
  • 桥接模式替代多层继承方案,可以减少子类的个数,降低系统的管理和维护成本
  • 桥接模式的引入增加了系统的理解和设计难度(较难分析出哪些是抽象层,哪些是实现层),由于聚合关联关系建立在抽象层要求开发者针对抽象进行设计和编程
  • 桥接模式要求正确识别出系统中两个独立变化的维度,因此其使用范围有一定的局限性,即需要有这样的应用场景

文章说明

  • 本文章为本人学习尚硅谷的学习笔记,文章中大部分内容来源于尚硅谷视频(点击学习尚硅谷相关课程),也有部分内容来自于自己的思考,发布文章是想帮助其他学习的人更方便地整理自己的笔记或者直接通过文章学习相关知识,如有侵权请联系删除,最后对尚硅谷的优质课程表示感谢。
  • 本人还同步阅读《图解设计模式》书籍(图解设计模式/(日)结城浩著;杨文轩译–北京:人民邮电出版社,2017.1),进而综合两者的内容,让知识点更加全面

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++第三方开发库matplotlib-cpp

Matplotlib-cpp是一个用于在C中绘制图表的开源库。它提供了与Python的Matplotlib库类似的功能&#xff0c;使得在C环境下进行数据可视化变得更加便捷。基于Matplotlib-cpp&#xff0c;我们可以使用各种绘图函数和样式选项来创建各种类型的图表&#xff0c;包括折线图、散点图、…

Qtday2作业

1. 头文件 widget.h #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QDebug> #include<QIcon> #include<QPushButton> #include<QLineEdit> #include<QLabel> #include<QMovie> class Widget : public QWidget…

【多线程带来的的风险-线程安全的问题的简单实例-线程不安全的原因】

文章目录 前言线程不安全的5大原因1. 抢占式执行和随机调度2. 多个线程同时修改一个变量(共享数据&#xff09;3. 修改操作不是原子性的4. 内存可见性5. 指令重排序 前言 什么是线程安全&#xff1f; 简单来说&#xff0c;如果多线程环境下代码运行的结果是符合我们预期的&am…

【SpringCloud Alibaba】(一)微服务介绍

此专栏内容皆来自于【冰河】的《SpringCloud Alibaba 实战》文档。 1. 专栏介绍 我们先来看看《SpringCloud Alibaba实战》专栏的整体结构吧&#xff0c;先上图 从上图&#xff0c;大家可以看到&#xff0c;专栏从整体上分为十个大的篇章&#xff0c;分别为 专栏设计、微服务…

macOS Ventura 13.5 (22G74) 正式版发布,ISO、IPSW、PKG 下载

macOS Ventura 13.5 (22G74) 正式版发布&#xff0c;ISO、IPSW、PKG 下载 本站下载的 macOS Ventura 软件包&#xff0c;既可以拖拽到 Applications&#xff08;应用程序&#xff09;下直接安装&#xff0c;也可以制作启动 U 盘安装&#xff0c;或者在虚拟机中启动安装。另外也…

全志F1C200S嵌入式驱动开发(GPIO输出)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】 和v3s一样,f1c200s本身的外部引脚比较少。所以这个时候,不可避免地,很多引脚的功能就会重叠在一起。这种情况下,我们就要学会取舍了。比如说,如果是学习sd卡的时候,那么spi的…

阿里云NVIDIA A100 GPU云服务器性能详解及租用费用

阿里云GPU服务器租用费用表包括包年包月、一个小时收费以及学生GPU服务器租用费用&#xff0c;阿里云GPU计算卡包括NVIDIA V100计算卡、T4计算卡、A10计算卡和A100计算卡&#xff0c;GPU云服务器gn6i可享受3折&#xff0c;阿里云百科分享阿里云GPU服务器租用表、GPU一个小时多少…

拦截Bean使用之前各个时机的Spring组件

拦截Bean使用之前各个时机的Spring组件 之前使用过的BeanPostProcessor就是在Bean实例化之后&#xff0c;注入属性值之前的时机。 Spring Bean的生命周期本次演示的是在Bean实例化之前的时机&#xff0c;使用BeanFactoryPostProcessor进行验证&#xff0c;以及在加载Bean之前进…

最新Ai创作源码ChatGPT商用运营源码/支持GPT4.0+支持ai绘画+支持Mind思维导图生成

本系统使用Nestjs和Vue3框架技术&#xff0c;持续集成AI能力到本系统&#xff01; 支持GPT3模型、GPT4模型Midjourney专业绘画&#xff08;全自定义调参&#xff09;、Midjourney以图生图、Dall-E2绘画Mind思维导图生成应用工作台&#xff08;Prompt&#xff09;AI绘画广场自定…

STM32—CAN通信

文章目录 一、CAN通信简介1.1 CAN简介1.2 CAN协议特点1.3 CAN通信的帧类型1.4 数据帧结构1.5 CAN的位时序1.6 CAN的仲裁功能 二、STM32F1的CAN2.1 bxCAN简介2.2 bxCAN工作模式2.2.1 初始化模式2.2.2 正常模式2.2.3 睡眠模式2.2.4 静默模式2.2.5 环回模式 2.3 位时序和波特率 三…

leetCode刷题记录3-面试经典150题

文章目录 不要摆&#xff0c;没事干就刷题&#xff0c;只有好处&#xff0c;没有坏处&#xff0c;实在不行&#xff0c;看看竞赛题面试经典 150 题80. 删除有序数组中的重复项 II189. 轮转数组122. 买卖股票的最佳时机 II 不要摆&#xff0c;没事干就刷题&#xff0c;只有好处&…

阿里云部署 ChatGLM2-6B 与 langchain+ChatGLM

1.ChatGLM2-6B 部署 更新系统 apt-get update 安装git apt-get install git-lfs git init git lfs install 克隆 ChatGLM2-6B 源码 git clone https://github.com/THUDM/ChatGLM2-6B.git 克隆 chatglm2-6b 模型 #进入目录 cd ChatGLM2-6B #创建目录 mkdir model #进入目录 cd m…

安全攻击 --- XSS攻击

XSS跨站脚本攻击 &#xff08;1&#xff09;简介 OWASP TOP 10 之一&#xff0c;XSS被称为跨站脚本攻击&#xff08;Cross-Site-Scripting&#xff09;主要基于JavaScript&#xff08;JS&#xff09;完成攻击行为XSS通过精心构造JS代码注入到网页中&#xff0c;并由浏览器解释…

在nginx上部署nuxt项目

先安装Node.js 我安的18.17.0。 安装完成后&#xff0c;可以使用cmd&#xff0c;winr然cmd进入&#xff0c;测试是否安装成功。安装在哪个盘都可以测试。 测试 输入node -v 和 npm -v&#xff0c;&#xff08;中间有空格&#xff09;出现下图版本提示就是完成了NodeJS的安装…

前端开发实习总结参考范文

▼前端开发实习总结篇四 读了三年的大学&#xff0c;然而大多数人对本专业的认识还是不那么透彻&#xff0c;学的东西真正能够学以致用的东西很少&#xff0c;大家都抱怨没有实践的机会&#xff0c;在很多同学心里面对于本专业还是很茫然。直到即将毕业的时候才知道我们以前学…

【Linux后端服务器开发】HTTPS协议

目录 一、加密算法 二、中间人攻击 三、CA认证 一、加密算法 HTTPS协议是什么&#xff1f;HTTPS协议也是一个应用层协议&#xff0c;是在HTTP协议的基础上引入了一个加密层。 HTTP协议内容是按照文本的方式明文传输的&#xff0c;这就导致在传输过程中出现一些被篡改的情况…

ROS1ROS2之CmakeList.txt和package.xml用法详解

前言&#xff1a;目前还在学习ROS无人机框架中&#xff0c;&#xff0c;&#xff0c; 更多更新文章详见我的个人博客主页【前往】 文章目录 1. CMakeLists.txt与package.xml的作用2. 生成CMakeLists.txt2.1 ROS12.2 ROS2 3. CMakeLists.txt编写3.1 ROS13.2 ROS2 4. package.xml…

Ubuntu 20.04 Ubuntu18.04安装录屏软件Kazam

1.在Ubuntu Software里面输入Kazam&#xff0c;就可以找不到这个软件&#xff0c;直接点击install就可以了 2.使用方法&#xff1a; 选择Screencast&#xff08;录屏&#xff09; Fullscreen&#xff08;全屏&#xff09;-----Windows&#xff08;窗口&#xff09;--------Ar…

1、传统锁回顾(Jvm本地锁,MySQL悲观锁、乐观锁)

目录 1.1 从减库存聊起1.2 环境准备1.3 简单实现减库存1.4 演示超卖现象1.5 jvm锁1.6 三种情况导致Jvm本地锁失效1、多例模式下&#xff0c;Jvm本地锁失效2、Spring的事务导致Jvm本地锁失效3、集群部署导致Jvm本地锁失效 1.7 mysql锁演示1.7.1、一个sql1.7.2、悲观锁1.7.3、乐观…

fragment

fragment 在vue2中,组件必须有一个跟标签在vue3中,组件可以没有跟标签,内部会将多个标签包含在一个fragment虚拟元素中好处:减少标签层级,减小内存占用 teltport 什么是teltport teleport是一种能够将我们组件html结构移动到指定位置的技术 像是下面的代码不适用teleport:…