【数据结构与算法——TypeScript】图结构(Graph)

【数据结构与算法——TypeScript】

图结构(Graph)

认识图结构以及特性

什么是图?

  • 在计算机程序设计中,图结构 也是一种非常常见的数据结构。
    • 但是,图论其实是一个非常大的话题
  • 认识一下关于图的一些内容
    • 图的抽象数据类型
    • 一些算法实现。
  • 什么是图?
    • 图结构是一种与树结构有些相似的数据结构。
    • 图论数学的一个分支,并且,在数学的概念上,树是图的一种。
    • 它以图为研究对象,研究 顶点 组成的图形的数学理论和方法
    • 主要研究的目的是事物之间的关系顶点代表事物代表两个事物间的关系
  • 我们知道树可以用来模拟很多现实的数据结构
    • 比如: 家谱/公司组织架构等等
  • 那么图长什么样子?
  • 或者什么样的数据使用图来模拟更合适呢?

图的现实案例

  • 人与人之间的关系网
    • 甚至科学家们在观察人与人之间的关系网时,还发现了六度空间理论
  • 六度空间理论
    • 理论上认为世界上任何两个互相不认识的两人。
    • 只需要很少的中间人就可以建立起联系。
    • 并非一定要经过6步,只是需要很少的步骤。
  • 地铁图

在这里插入图片描述

  • 村庄之间的关系网

在这里插入图片描述

再次 什么是图?

  • ❤️‍🔥 那么,什么是图呢?
    • 我们会发现,上面的节点(其实图中叫顶点Vertex)之间的关系,是不能使用树来表示
    • 使用任何的树结构都不可以模拟。
    • 这个时候,我们就可以使用图来模拟它们。
  • ❤️‍🔥 图通常有什么特点呢?
    • 💚 一组顶点:通常用 V (Vertex) 表示顶点的集合
    • 💚 一组边:通常用 E (Edge) 表示边的集合
      • ✓ 边是顶点和顶点之间的连线
      • ✓ 边可以是有向的,也可以是无向的。
      • ✓ 比如A — B,通常表示无向。 A --> B,通常表示有向

欧拉和七拉问题解法

历史故事

  • 8世纪著名古典数学问题之一。
    • 在哥尼斯堡的一个公园里,有七座桥普雷格尔河中两个岛岛与河岸连接起来(如图)。

      在这里插入图片描述

    • 有人提出问题: 一个人怎样才能不重复、不遗漏地一次走完七座桥,最后回到出发点。

  • 1735年,有几名大学生写信给当时正在俄罗斯的彼得斯堡科学院任职的瑞典天才数学家欧拉,请他帮忙解决这一问题。
    • 欧拉在亲自观察了哥伦斯堡的七桥后,认真思考走法,但是始终没有成功,于是他怀疑七桥问题是不是无解的。
    • 1736年29岁的欧拉向 彼得斯堡 科学院递交了《哥尼斯堡的七座桥》的论文,在解答问题的同时,开创了数学的一个新的分支——图论与几何拓扑,也由此展开了数学史上的新历程。

欧拉解答

  • 他不仅解决了该问题,并且给出了 连通图 可以一笔画的充要条件是:
    • 奇点的数目不是0个就是2 个
    • 连到一点的边的数目如果是奇数条,就称为奇点
    • 如果是偶数条就称为偶点
    • 要想一笔画成,必须中间点均是偶点
    • 也就是有来路必有另一条去路奇点只可能在两端,因此任何图能一笔画成,奇点要么没有,要么在两端
  • 个人思考:
    • 欧拉在思考这个问题的时候,并不是针对某一个特性的问题去考虑,而是将岛和桥抽象成了点和线
    • 抽象是数学的本质,而编程我们也一再强调抽象的重要性。
    • 汇编语言是对机器语言的抽象,高级语言是对汇编语言的抽象。
    • 操作系统是对硬件的抽象,应用程序在操作系统的基础上构建。

在这里插入图片描述

图结构的常见术语

关于术语的概述

  • 我们在学习树的时候,树有很多的相关术语

  • 了解这些术语有助于我们更好的理解树结构

  • 我们也来学习一下图相关的术语

    • 但是图的术语其实非常多,如果你找一本专门讲图的各个方面的书籍,会发现 只是术语 就可以 占据满满的一个章节。
    • 这里,我们先介绍几个比较常见的术语。
  • 我们先来看一个抽象出来的图
     用数字更容易我们从整体来观察整个图结构
    在这里插入图片描述

术语

  1. 顶点:

    • 顶点刚才我们已经介绍过了,表示图中的一个节点
    • 比如地铁站中某个站/多个村庄中的某个村庄/互联网中的某台主机/人际关系中的人
  2. 边:

    • 边刚才我们也介绍过了,表示顶点和顶点之间的连线
    • 比如地铁站中两个站点之间的直接连线,就是一个边。
    • ❗️ 注意: 这里的边不要叫做路径,路径有其他的概念,待会儿我们会介绍到。
    • 之前的图中: 0 - 1有一条边,1 - 2有一条边,0 - 2没有边。
  3. 相邻顶点:

    • 由一条边连接在一起的顶点称为相邻顶点
    • 比如0 - 1是相邻的,0 - 3是相邻的。 0 - 2是不相邻的。
  4. 度:

    • 一个顶点的度是相邻顶点的数量。
    • 比如0顶点和其他两个顶点相连,0顶点的度是2
    • 比如1顶点和其他四个顶点相连,1顶点的度是4
  5. 路径:

    • 路径是顶点v1,v2…,vn的一个连续序列,比如上图中0 1 5 9就是一条路径。
    • 简单路径: 简单路径要求不包含重复的顶点。 比如 0 1 5 9 是一条简单路径。
    • 回路: 第一个顶点和最后一个顶点相同的路径称为回路。 比如 0 1 5 6 3 0
  6. 无向图:

    • 上面的图就是一张无向图,因为所有的边都没有方向
    • 比如 0 - 1之间有变,那么说明这条边可以保证 0 -> 1,也可以保证 1 -> 0。
  7. 有向图:

    • 有向图表示的图中的边是有方向的。
    • 比如 0 -> 1,不能保证一定可以 1 -> 0,要根据方
      向来定。

在这里插入图片描述

  1. 无权图:

    • 我们上面的图就是一张无权图(边没有携带权重)
    • 我们上面的图中的边是没有任何意义
    • 不能说 0 - 1的边,比4 - 9的边更远或者用的时间更长。
  2. 带权图:

    • 带权图表示边有一定的权重
    • 这里的权重可以是任意你希望表示的数据
      ✓ 比如距离或者花费的时间或者票价

图的表示

  • 怎么在程序中表示图呢?
    • 我们知道一个图包含很多顶点,另外包含顶点和顶点之间的连线(边)
    • 这两个都是非常重要的图信息,因此都需要在程序中体现出来
  • 顶点的表示相对简单,我们先讨论顶点的表示。
    • 上面的顶点,我们抽象成了1 2 3 4,也可以抽象成A B C D。
    • 在后面的案例中,我们使用A B C D。
    • 那么这些A B C D我们可以使用一个数组来存储起来(存储所有的顶点)
    • 当然,A,B,C,D也可以表示其他含义的数据(比如村庄的名字).
  • 那么边怎么表示呢?
    • 因为边是两个顶点之间的关系,所以表示起来会稍微麻烦一些。
    • 下面,我们具体讨论一下变常见的表示方式。

邻接矩阵和邻接表

邻接矩阵

  • 一种比较常见的表示图的方式: 邻接矩阵
    • 邻接矩阵让每个节点和一个整数项关联,该整数作为数组的下标值
    • 我们用一个二维数组来表示顶点之间的连接
    • 二维数组 [0][2] -> A -> C
  • 画图演示:

在这里插入图片描述

  • 图片解析:
    • 在二维数组中,0表示没有连线,1表示有连线。
    • 通过二维数组,我们可以很快的找到一个顶点和哪些顶点有连线。(比如A顶点,只需要遍历第一行即可)
    • 另外,A - A,B - B(也就是顶点到自己的连线),通常使用0表示。
  • 邻接矩阵的问题:
    • 邻接矩阵还有一个比较严重的问题,就是如果图是一个稀疏图
    • 那么矩阵中将存在大量的0,这意味着我们浪费了计算机存储空间来表示根本不存在的边

邻接表

  • 另外一种常用的表示图的方式: 邻接表

    • 邻接表由图中每个顶点以及和顶点相邻的顶点列表组成。
    • 这个列表有很多种方式来存储: **数组/链表/字典(哈希表)**都可以。
  • 画图演示
    在这里插入图片描述

  • 图片解析:

    • 其实图片比较容易理解。
    • 比如我们要表示和A顶点有关联的顶点(边),A和B/C/D有边,
    • 那么我们可以通过A找到对应的数组/链表/字典,再取出其中的内容就可以啦。
  • 邻接表的问题:

    • 邻接表计算"出度"是比较简单的(出度: 指向别人的数量,入度: 指向自己的数量)
    • 邻接表如果需要计算有向图的"入度",那么是一件非常麻烦的事情。
    • 它必须构造一个**“逆邻接表”,才能有效的计算“入度”。但是开发中“入度”**相对用的比较少。

创建图类

  • 先创建Graph类
  • 👩🏻‍💻 代码解析
    • 创建Graph的构造函数,这个我们在封装其他数据结构的时候已经非常熟悉了。
  • 定义了两个属性:
    ✓ vertexes: 用于存储所有的顶点,我们说过使用一个数组来保存。
    ✓ adjList: adj是adjoin的缩写,邻接的意思。 adjList用于存储所有的边,我们这里采用邻接表的形式。
  • 之后,我们来定义一些方法以及实现一些算法就是一个完整的图类了。
    class Graph<T> {
    // 顶点
    vertexes: T[] = [];
    // 邻接表
    adjList: Map<T, T[]> = new Map();
    }

添加方法

  • 现在我们来增加一些添加方法。
    • 添加顶点: 可以向图中添加一些顶点。
    • 添加边: 可以指定顶点和顶点之间的边。
  • 添加顶点
    • 👩🏻‍💻 代码解析:
      • 我们将添加的顶点放入到数组中。
      • 另外,我们给该顶点创建一个数组[],该数组用于存储顶点连接的所有的边.(回顾邻接表的实现方式)
  • 添加边
    • 👩🏻‍💻 代码解析:
      • 添加边需要传入两个顶点,因为边是两个顶点之间的边,边不可能单独存在。
      • 根据顶点v取出对应的数组,将w加入到它的数组中。
      • 根据顶点w取出对应的数组,将v加入到它的数组中。
      • 因为我们这里实现的是无向图,所以边是可以双向的。
  // 添加顶点
  addVertex(vertex: T) {
    //   将顶点添加到顶点数组中保存
    this.vertexes.push(vertex);
    //   创建一个邻接表的数组
    this.adjList.set(vertex, []);
  }
  addEdge(v1: T, v2: T) {
    this.adjList.get(v1)?.push(v2);
    this.adjList.get(v2)?.push(v1);
  }

printEdges方法

  • 为了能够正确的显示图的结果,我们来实现一下Graph的printEdges方法

  • 测试代码:

      const graph = new Graph();
    
      graph.addVertex('A');
      graph.addVertex('B');
      graph.addVertex('C');
      graph.addVertex('D');
      graph.addVertex('E');
      graph.addVertex('F');
      graph.addVertex('G');
      graph.addVertex('H');
      graph.addVertex('I');
    
      graph.addEdge('A', 'B');
      graph.addEdge('A', 'C');
      graph.addEdge('A', 'D');
      graph.addEdge('C', 'D');
      graph.addEdge('C', 'G');
      graph.addEdge('D', 'G');
      graph.addEdge('D', 'H');
      graph.addEdge('B', 'E');
      graph.addEdge('B', 'F');
      graph.addEdge('E', 'I');
    
      graph.printEdges();
    
    
    • 运行结果:
      [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fK63qbI7-1692005552724)(/Volumes/web/数据结构与算法(ts)/截图/截屏2023-08-14 15.37.06.png “运行结果”)]
      在这里插入图片描述

图的遍历

图的遍历思想

图的遍历思想和树的遍历思想是一样的。
图的遍历意味着需要将图中每个顶点访问一遍,并且不能有重复的访问

  • 有两种算法可以对图进行遍历
    • 广度优先搜索(Breadth-First Search,简称BFS)
    • 深度优先搜索(Depth-First Search,简称DFS)
  • 两种遍历算法,都需要明确指定第一个被访问的顶点
    • 它们的遍历过程分别是怎么样呢?
    • 我们以一个迷宫中关灯为例。
    • 现在需要你进入迷宫,将迷宫中的灯一个个关掉,你会怎么关呢?

遍历的思想

  • 两种算法的思想:
    • BFS: 基于队列,入队列的顶点先被探索。
    • DFS: 基于栈或使用递归,通过将顶点存入栈中,顶点是沿着路径被探索的,存在新的相邻顶点就去访问。
  • 为了记录顶点是否被访问过,我们使用三种颜色来反应它们的状态
    • 白色: 表示该顶点还没有被访问。
    • 灰色: 表示该顶点被访问过,但并未被探索过。
    • 黑色: 表示该顶点被访问过且被完全探索过。
  • 或者我们也可以使用Set来存储被访问过的节点。

广度优先搜索

  • 广度优先搜索算法的思路 💡:

    • 广度优先算法会从指定的第一个顶点开始遍历图,先访问其所有的相邻点,就像一次访问图的一层
    • 换句话说,就是先宽后深的访问顶点
  • 图解BFS
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mNFb1uIZ-1692005552726)(/Volumes/web/数据结构与算法(ts)/画图/图/广度优先搜索.png “广度优先搜索”)]
    在这里插入图片描述

    ◼ 广度优先搜索的实现:
    ◼ 创建一个队列Q。

  • 代码 💻 :

  // 广度优先
  bfs() {
    // 1. 判断是否有顶点
    if (this.vertexes.length === 0) return;
    // 2. 创建队列结构来访问每个顶点
    const queue: T[] = [];
    queue.push(this.vertexes[0]);
    // 3. 创建Set,来记录一个顶点是否被访问过
    const visited = new Set();
    visited.add(this.vertexes[0]);
    // 4. 遍历队列中每一个顶点
    while (queue.length) {
      //  4.1 访问队列中第一个顶点
      const vertex = queue.shift()!;
      console.log(vertex);
      //  4.2 取出相邻的顶点
      const neighbors = this.adjList.get(vertex);
      if (!neighbors) continue;
      for (const neighbor of neighbors) {
        if (!visited.has(neighbor)) {
          visited.add(neighbor);
          queue.push(neighbor);
        }
      }
    }
  }
  • 运行结果:

在这里插入图片描述

深度优先搜索

  • 深度优先搜索的思路:

    • 深度优先搜索算法将会从第一个指定的顶点开始遍历图,沿着路径知道这条路径最后被访问了。
    • 接着原路回退并探索下一条路径。
  • 深度优先搜索

    • 深度优先搜索算法的实现:
      • 广度优先搜索算法我们使用的是队列,这里可以使用完成,也可以使用递归。
  • 图DFS

    在这里插入图片描述

  • 代码 💻 :

  // 深度优先
  dfs() {
    // 1. 判断有没有顶点,没有直接返回
    if (this.vertexes.length === 0) return;
    // 2.创建栈结构
    const stack: T[] = [];
    stack.push(this.vertexes[0]);

    // 3. 创建Set结构
    const visited = new Set();
    visited.add(this.vertexes[0]);
    // 4. 从第一个顶点开始访问
    while (stack.length) {
      const vertex = stack.pop()!;
      console.log(vertex);
      // 取出相邻的顶点
      const neighbors = this.adjList.get(vertex);
      if (!neighbors) continue;
      for (let i = neighbors.length - 1; i >= 0; i--) {
        const nei = neighbors[i];
        if (!visited.has(nei)) {
          visited.add(nei);
          stack.push(nei);
        }
      }
    }
  }
  • 运行结果:

在这里插入图片描述

图结构的常见建模

  • 对交通流量建模
    • 顶点可以表示街道的十字路口,边可以表示街道。
    • 加权的边可以表示限速或者车道的数量或者街道的距离。
    • 建模人员可以用这个系统来判定最佳路线以及最可能堵车的街道。
  • 对飞机航线建模
    • 航空公司可以用图来为其飞行系统建模。
    • 将每个机场看成顶点,将经过两个顶点的每条航线看作一条边。
    • 加权的边可以表示从一个机场到另一个机场的航班成本,或两个机场间的距离。
    • 建模人员可以利用这个系统有效的判断从一个城市到另一个城市的最小航行成本。

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

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

相关文章

一百五十五、Kettle——Linux上安装的kettle9.3连接MySQL数据库

一、目的 kettle9.3在Linux上成功安装后&#xff0c;就建立数据库连接&#xff0c;第一个就是MySQL数据库 二、前提准备 提前准备好MySQL驱动包 &#xff08;一&#xff09;MySQL版本 &#xff08;二&#xff09;注意&#xff1a;由于我的MySQL版本比较高&#xff0c;所以特…

maven工具-maven的使用-镜像仓库、本地仓、IDEA使用maven

Maven 一、为什么使用maven 添加第三方jar包jar包之间的依赖关系处理jar包之间的冲突获取第三方jar包将项目拆分成多个工程模块实现项目的分布式部署 二、maven简介 ​ Maven项目对象模型(POM)&#xff0c;可以通过一小段描述信息来管理项目的构建&#xff0c;报告和文档的…

动手学深度学习-pytorch版本(一):引言 预备知识

参考引用 动手学深度学习利用 Anaconda 安装 pytorch 和 paddle 深度学习环境 pycharm 安装 0. 环境安装 利用 Anaconda 安装 pytorch 和 paddle 深度学习环境 pycharm 安装 1. 引言 机器学习&#xff08;machine learning&#xff0c;ML&#xff09;是⼀类强⼤的可以从经…

数据结构:力扣刷题

题一&#xff1a;旋转数组 给定一个整数数组 nums&#xff0c;将数组中的元素向右轮转 k 个位置&#xff0c;其中 k 是非负数。 思路一&#xff1a; 创建reverse()函数传入三个值分别为数组地址&#xff0c;从第几个数组元素开始&#xff0c;结束元素位置&#xff1b; 在r…

性能场景和性能需求指标

目录 一 性能场景 1、基准性能场景 2、容量性能场景 3、稳定性性能场景 4、异常性能场景 二 性能需求指标 1、业务指标 2、技术指标 2.1 时间指标 RT 2.2 容量指标 TPS 2.3 资源利用率 3、指标之间的关系 “TPS”与“响应时间” “用户数”与“TPS”与“压力工具中…

设计HTML5表格

在网页设计中&#xff0c;表格主要用于显示包含行、列结构的二维数据&#xff0c;如财务表格、调查数据、日历表、时刻表、节目表等。在大多数情况下&#xff0c;这类信息都由列标题或行标题及数据构成。本章将详细介绍表格在网页设计中的应用&#xff0c;包括设计符合标准化的…

构建之法 - 软件工程实践教学:每天都向前推进一点点

作者&#xff1a;福州⼤学 汪璟玢⽼师 汪老师&#xff1a;每次都向前推进一点点&#xff0c;哪怕只有一点点&#xff0c;也好过什么都不做。 ​邹老师&#xff1a;对&#xff0c;几个学期下来&#xff0c;就已经超过那些“空想”的团队很远了。坚持下去&#xff01; 汪老师&…

IEEE权威杂志特别报道一款国内大语言模型,不是百度、不是阿里、不是华为……

现如今&#xff0c;大模型早已成为街谈巷议的 AI 话题&#xff0c;ChatGPT 更火出了圈。NLP、CV 和多模态领域的大模型层出不穷&#xff0c;并朝着通用方向不断演进&#xff0c;让人们看到了 AGI 的希望。 在大模型时代&#xff0c;除了继续升级预训练、微调等技术和算法之外&a…

武汉地铁19号线完成5G专网全覆盖,现场测试下行速率超千兆!

近日&#xff0c;极目新闻记者从中国移动湖北公司获悉&#xff0c;随着武汉地铁19号线全线隧道正式贯通&#xff0c;湖北移动目前已完成新月溪公园至鼓架山站5G网络覆盖&#xff0c;轨行区5G专网全覆盖&#xff0c;并成功进行试车验证&#xff0c;19号线成为国内首条全线实现5G…

基于ipad协议的gewe框架进行微信群组管理(二)

友情链接 geweapi.com 点击访问即可。 获取群组详情 小提示&#xff1a; 该接口可以一次查询20个群组查询出来的信息是不带公告的 请求URL&#xff1a; http://域名地址/api/group/detail 请求方式&#xff1a; POST 请求头&#xff1a; Content-Type&#xff1a;applica…

Base64编码-算法特别的理解

Base64 在DES加密和AES加密的过程中&#xff0c;加密的编码会出现负数&#xff0c;在ascii码表中找不到对应的字符&#xff0c;就会出现乱码。为了解决乱码的问题&#xff0c;一般结合base64使用 所谓Base64&#xff0c;即是说在编码过程中使用了64种字符&#xff1a;大写A到Z、…

数据结构:力扣OJ题(每日一练)

题一&#xff1a;有效的括号 给定一个只包括 (&#xff0c;)&#xff0c;{&#xff0c;}&#xff0c;[&#xff0c;] 的字符串 s &#xff0c;判断字符串是否有效。 有效字符串需满足&#xff1a; 左括号必须用相同类型的右括号闭合。左括号必须以正确的顺序闭合。每个右括号…

智安网络|恶意软件在网络安全中的危害与应对策略

恶意软件是指一类具有恶意目的的软件程序&#xff0c;恶意软件是网络安全领域中的一个严重威胁&#xff0c;给个人用户、企业和整个网络生态带来巨大的危害。通过潜伏于合法软件、邮件附件、下载链接等途径传播&#xff0c;破坏用户计算机系统、窃取敏感信息、进行勒索等不法行…

Linux 终端操作命令(2)内部命令

Linux 终端操作命令 也称Shell命令&#xff0c;是用户与操作系统内核进行交互的命令解释器&#xff0c;它接收用户输入的命令并将其传递给操作系统进行执行&#xff0c;可分为内部命令和外部命令。内部命令是Shell程序的一部分&#xff0c;而外部命令是独立于Shell的可执行程序…

华为AI战略的CANN

基于TVM的华为昇腾体系中—— 异构计算架构&#xff08;CANN&#xff09;是对标英伟达的CUDA CuDNN的核心软件层&#xff0c;向上支持多种AI框架&#xff0c;向下服务AI处理器&#xff0c;发挥承上启下的关键作用&#xff0c;是提升昇腾AI处理器计算效率的关键平台 主要包括有…

CSS前端开发指南:创造精美的用户界面

简介&#xff1a; 《CSS前端开发指南&#xff1a;创造精美的用户界面》是一本旨在帮助读者掌握CSS技术&#xff0c;实现令人惊叹的前端用户界面的实用指南。无论您是初学者还是有经验的开发者&#xff0c;本书都将为您提供全面的知识和实用技巧&#xff0c;帮助您创建引人注目…

网页显示摄像头数据的方法---基于web video server

1. 背景&#xff1a; 在ros系统中有发布摄像头的相关驱动rgb数据&#xff0c;需求端需要将rgb数据可以直接在网页上去显示。 问题解决&#xff1a; web_video_server功能包&#xff0c;相关链接&#xff1a; web_video_server - ROS Wiki 2. 下载&#xff0c;安装和编译&a…

Java算法_ 二叉树的最大深度(LeetCode_Hot100)

题目描述&#xff1a;给定一个二叉树 &#xff0c;返回其最大深度。root 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 获得更多&#xff1f;算法思路:代码文档&#xff0c;算法解析的私得。 运行效果 完整代码 /*** 2 * Author: LJJ* 3 * Date: 2023/…

Rx.NET in Action 第四章学习笔记

Part 2 核心思想 《Rx.NET in Action》这一部共分八章&#xff0c;涵盖了Rx 关键模块——**Observable(可观察序列)和Observer(观察者)**的全部功能&#xff0c;以及如何创建它们、连接它们和控制它们之间的关系。 然后&#xff0c;您将学习如何使用强大的 Rx 处理器构建复杂…

2023年京东按摩仪行业数据分析(京东销售数据分析)

近年来&#xff0c;小家电行业凭借功能与颜值&#xff0c;取代黑电和白电&#xff0c;成为家电市场的主要增长点。在这一市场背景下&#xff0c;颜值更高、功能更丰富、品种更齐全的各类按摩仪&#xff0c;借助新消费和电子商务的风潮&#xff0c;陆续被推上市场。今年&#xf…