应用图扑 HT for Web 搭建拓扑关系图

图片

拓扑结构在计算机网络设计和通信领域中非常重要,因为它描述了网络中的设备(即“点”)如何相互连接(即通过“线”)。这种结构不仅涉及物理布局,即物理拓扑,还可以涉及逻辑或虚拟的连接方式,即逻辑拓扑

图片

图片

图扑软件自研 HT for Web 产品曾参与搭建了众多拓扑可视化解决方案。如机房通信拓扑可视化,实现通过图形图像直观展示机房内部网络设备、服务器、存储设备以及之间连接关系的技术。帮助 IT 管理员和网络工程师更加直观地理解机房的网络结构,便于故障排查、网络优化和规划扩展。

在 HT 中,ht.Node 可充当拓扑图中的“点”角色,ht.Node 上可显示图片图标,这使得创建拓扑图时能够直观地表示每一个“点”的特征。至于拓扑图中的“线”,即用于体现两个节点之间关系的元素,可由 ht.Edge 类型承担这一任务。ht.Edge用于连接起始和目标两个节点,两个节点间可以有多条ht.Edge存在,也允许起始和目标为同一节点。

在搭建 HT 拓扑图前,我们需要先创建一个 HT 的 2D 视图:

const dm = new ht.DataModel(); // 创建一个数据模型const g2d = new ht.graph.GraphView(dm); // 创建一个 2D 视图g2d.addToDOM(); // 将 2D 视图添加到 body 中dm.setBackground('rgb(240,237,237)'); // 设置背景

2D 视图还可按照需求开启树层次渲染:

dm.setHierarchicalRendering(true);

2D 视图创建完成之后就可以创建“点”和“线”了:

const node1 = createNode('symbols/电信/icon_交换机.json', { x: 0, y: 0 }, "交换机");const node2 = createNode('symbols/电信/icon_路由.json', { x: 300, y: 0 }, "路由");createEdge(node1, node2);function createNode(icon, position, name) {     const node = new ht.Node(); // 创建一个 ht.Node 节点     node.s({         'label': name,         'label.color': "#fff"     });     node.setImage(icon);     node.p(position);     node.setSize({ width: 100, height: 100 });     dm.add(node); // 将节点加到数据模型中     return node;}function createEdge(source, target, color, reverse) {     const edge = new ht.Edge(source, target); // 创建一个 ht.Edge     dm.add(edge); // 将连线节点加到数据模型中     return edge;}

运行代码后的效果:

图片

复杂连线

以上展示了一个简洁的示例,直观地展现了如何在图扑自研 HT for Web 中创建节点并将它们通过连线相连。实际应用中的情形往往更为复杂,需要根据实际数据构建拓扑结构。在这个过程中,核心步骤依旧是首先创建 ht.Node 实例以表示各个节点,再利用 ht.Edge 实例来实现节点之间的连接。接下来,让我们通过复杂一些的示例来演示。

创建节点

为了批量创建节点并方便管理节点数据,示例中定义了结构化的数据格式,并将数据存储在一个 JSON 文件中,再通过 ht.Default.xhrLoad() 去获取到 JSON 文件中的数据。获取到数据后就可批量创建节点。

在实际的运用场景中,也可以通过任何 web 端通讯方式 HTTP/AJAX、WebSocket 去获取数据。

equipment.json 文件中定义的数据格式:

[    {        "name": "核心交换机1", // 设备名称        "code": "EQ_ASBB1425", // 设备编码(唯一标识)        "icon": "symbols/user/900-word/电信拓扑图标/icon_核心交换机.json", // 设备 icon// 设备在图纸中的位置"size": 60, // 节点大小        "position": {            "x": 0,            "y": 100        }    },    {        "name": "核心交换机2",        "code": "EQ_ASBB1478",        "icon": "symbols/user/900-word/电信拓扑图标/icon_核心交换机.json",        "position": {            "x": 200,            "y": 0        }    },    {        "name": "服务器1",        "code": "EQ_BCGJ2121",        "icon": "symbols/user/900-word/电信拓扑图标/空白服务器.json",        "position": {            "x": 200,            "y": 250        }    },   ...]

获取到数据并批量创建节点:

ht.Default.xhrLoad('./equipment.json', function (json) {       const data = ht.Default.parse(json);       data.forEach((item) => {             createNode(item);        })})function createNode(data) {        const node = new ht.Node();        node.setTag(data.code); // 设置节点唯一标识        node.setImage(data.icon);        node.p(data.position);        node.s('2d.movable', false); // 禁止移动        node.setSize({ width: data.size || 150, height: data.size || 150 });        dm.add(node);         return node;}

图片

创建连线

与节点数据相同,示例中定义了连线对应格式,并且也是存储在一个 JSON 文件中,再通过 ht.Default.xhrLoad() 获取数据。JSON 文件中定义了连线中最重要的几个因素:起始节点、目标节点、连线颜色。

[    {        "source": "EQ_ASBB1425", // 起始节点的唯一标识        "target": "EQ_BCGJ2121", // 结束节点的唯一标识        "color": "rgb(0,199,7)" // 连线颜色    },    {        "source": "EQ_ASBB1425",        "target": "EQ_BCGJ2131",        "color": "rgb(0,199,7)"},...]

获取数据并且批量创建连线,这一步需要在创建节点之后执行:

ht.Default.xhrLoad('./connectData.json', function (json) {      const connectData = ht.Default.parse(json);      connectData.forEach((item) => {            createEdge(item);      })})function createEdge(data) {    const source = dm.getDataByTag(data.source);    const target = dm.getDataByTag(data.target);    const edge = new ht.Edge(source, target);    edge.s({        "edge.color": data.color || "rgb(0,199,7)",        "edge.width": 4,        "shadow2.offset.x": -4,        "shadow2.offset.y": 7,        "shadow2": true,        "shadow2.color": "rgba(0,0,0,0.18)",})dm.add(edge);    dm.moveToTop(edge); // 将节点移动至顶部    return edge;}

图片

到这里,基本上整个拓扑的效果都已经展示出来了,但是可能还存在一些问题。如终端路由之间的连线被服务器挡住了,可能会被认为是路由 1—服务器 1—服务器 2—路由 2 这样的连接。

图片

这种情况下,就可以采用其他的连线方式。ht.Edge 提供了多种的连线方式,可以通过 edge.s(‘edge.type’, 连接方式) 进行设置。下面展示几种不同的连接方式:

1.弯曲:edge.s(‘edge.type’, ‘flex2’)

图片

2.正交:edge.s(‘edge.type’, ‘ortho2’)

图片

3.先水平后垂直:edge.s(‘edge.type’, ‘h.v2’)

图片

4.先水平后垂直:edge.s(‘edge.type’, ‘v.h2’)

图片

......

ht.Edge 还有很多种的连线方式,这里就先介绍以上几种方式。

在这个示例内,两个路由之间的连线需要跨域多个其他的连线,为了使得连线更加美观易懂,于是我就这条连线采用了 points 的连线方式,这种方式有极高的灵活性,可在连线路径上自由地添加控制点,从而实现非常多样化的效果。

points 类型的连线,有两个非常重要的属性:

  • edge.points:控制点信息;

  • edge.segments:用来标识在绘制时如何使用 points 数组中的顶点信息。

将示例中这条连线的连线类型改为 points,并设置上相应的属性:

edge.s({    'edge.type': 'points',    'edge.center': true,    'edge.points': [        { "x": 680, "y": 105 },        { "x": 490, "y": 200 },        { "x": 470, "y": 200 },        { "x": 410, "y": 230 },        { "x": 400, "y": 250 },        { "x": 360, "y": 270 },        { "x": 340, "y": 270 },        { "x": 260, "y": 310 },        { "x": 250, "y": 330 },        { "x": 80, "y": 415 }    ],    'edge.segments': [1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2]});

使用以上 points 连线类型实现的效果:

图片

增加背景和装饰

可创建一些 ht.Shape 节点作为背景装饰,突出显示特定的设备区域。

createShape([    { x: -100, y: 130 },    { x: 230, y: -50 },    { x: 340, y: 15 },    { x: 10, y: 195 },    { x: -100, y: 130 }], [1, 2, 2, 2, 5])createShape([    { x: -155, y: 354 },    { x: 575, y: -30 },    { x: 805, y: 110 },    { x: 60, y: 510 },    { x: -155, y: 355 },], [1, 2, 2, 2, 5]);createShape([    { x: 300, y: 470 },    { x: 660, y: 275 },    { x: 805, y: 350 },    { x: 435, y: 550 },    { x: 300, y: 470 },], [1, 2, 2, 2, 5])function createShape(points, segments) {    const shape = new ht.Shape();    shape.setPoints(points);    shape.setSegments(segments);    shape.s({        'shape.background': "#fff",        "shape.border.color": "rgba(13,46,79,0.67)",        "shape.border.width": 0.5,    })    dm.add(shape);    dm.moveToTop(shape);    return shape;}

添加背景后的效果如下:

图片

增加一些装饰的节点,这些节点本质上也都是 ht.Node,只是显示了不同的图标/图片,效果如下:

图片

添加箭头

在复杂的网络拓扑中,连线上常常需要表示数据流动方向。在使用图扑 HT 绘制连线时,ht.Edge 提供了 icons 属性,通过 icons 属性,可在 ht.Edge 上定义一系列图标并设置它们在连线上的位置。

在设置 icons 属性前,需要先注册好图标:

ht.Default.setImage('toArrow', {    width: 40,    height: 20,    comps: [        {            type: 'shape',points: [5, 2, 10, 10, 5, 18, 20, 10],            closePath: true,            background: 'rgb(0,199,7)',            borderWidth: 1,            borderColor: 'rgb(0,199,7)',            gradient: 'spread.vertical'        }    ]});ht.Default.setImage('fromArrow', {    width: 12,    height: 12,    comps: [        {            type: 'circle',            rect: [1, 1, 10, 10],            background: 'rgb(0,199,7)'        }    ]});

在 ht.Edge 上设置 icons:

edge.addStyleIcon("fromArrow", {     position: 15, // 图标位置     keepOrien: true, // 图标是否默认自动调整方向以保持最好的阅读效果     names: ['fromArrow']});edge.addStyleIcon("toArrow", {     position: 19,     keepOrien: true,     names: ['toArrow']});

设置 icons 之后的效果:

图片

流动动画

在图扑自研产品 HT for Web 中,使用 ht-flow.js 插件,能够为 ht.Edge 连线添加流动动画效果。这种效果可用于表示数据传输、能源流动或任何类型的动态连接。使用 ht-flow.js 插件实现的流动效果配置起来也十分简单,正确引入 ht-flow.js 插件后,使用 g2d.enableFlow(60); 开启流动,再在 ht.Edge 上设置相应的流动属性即可。

ht.Edge 配置流动效果的一些属性说明:

  • flow:布尔值,设置为 true 以启用流动效果。

  • flow.count:控制流动组的个数,默认为 1。

  • flow.step:控制流动的步进,默认为 3。

  • flow.element.count: 每个流动组中的元素的个数,默认为 10。

  • flow.element.space: 流动组中元素的间隔,默认为 3.5。

  • flow.element.image: 字符串类型,指定流动组中元素的图片,图片需要提前通过 ht.Default.setImage 注册。目前支持设置。

  • flow.element.background: 流动组中元素的背景颜色,默认为 rgba(255, 255, 114, 0.4)。

  • flow.element.shadow.begincolor: 字符串类型,表示流动组中的元素的渐变阴影的中心颜色,默认为 rgba(255, 255, 0, 0.3)。

  • flow.element.shadow.endcolor: 字符串类型,表示流动组中的元素的渐变阴影的边缘颜色,默认为 rgba(255, 255, 0, 0)。

  • flow.element.shadow.visible:流动阴影是否可见。

  • flow.begin.percent:开始的位置,值为 0 - 1,默认是 0。

  • flow.element.autorotate:是否自动朝向,根据连线的角度自动朝向。

在示例的 ht.Edge 上设置流动属性:

edge.s({        "flow": true,        "flow.element.background": "rgba(240, 225, 19, 0.5)",        "flow.element.shadow.begincolor": "rgba(240, 225, 19, 0.5)",        "flow.element.shadow.endcolor": "rgba(240, 225, 19, 0)",        "flow.element.count": 1 });

设置完成后的效果:

图片

在更为复杂的场景中,仅仅依赖简单的样式配置难以满足设计需求,为此 ht-flow.js 提供了 flow.element.image 属性,该属性支持将流动的元素设置为图片或图标,还支持设置为多个图片/图标流动的效果。

在流动上设置图标,需要先注册图标:

ht.Default.setImage('dataIcon1', {    "width": 50,    "height": 50,    "comps": [        {            "type": "shape",            "background": "rgb(125,195,125)",            "borderColor": "#979797",            "points": [                2.94441,                16.1039,                26.41008,                16.1039,                26.41008,                4.28571,                47.05559,                25.58442,                27.23783,                45.71429,                27.23783,                33.84863,                2.94441,                33.84863,                2.94441,                16.1039            ]        }    ]})ht.Default.setImage('dataIcon2', {    "width": 50,    "height": 50,    "comps": [        {            "type": "shape",            "background": "#32D3EB",            "borderColor": "#979797",            "points": [                2.94441,                16.1039,                26.41008,                16.1039,                26.41008,                4.28571,                47.05559,                25.58442,                27.23783,                45.71429,                27.23783,                33.84863,                2.94441,                33.84863,                2.94441,                16.1039            ]        }    ]});在 ht.Edge 上设置属性:edge.s({    "flow": true,    "flow.element.count": 2,    "flow.element.image": ["dataIcon1", "dataIcon2"],    "flow.element.max": 20,    "flow.element.min": 20,    "flow.element.shadow.visible": false,    "flow.element.space": 50,    "flow.element.autorotate": true});

设置完成后的效果:

图片

拓扑可视化优点

  1. 直观性:将抽象的关系和数据通过图形呈现,使得人们可以直观地理解和分析系统或网络的结构。

  2. 互动性:现代拓扑可视化工具通常支持用户与图形的交互操作,如缩放、拖拽节点、探索节点之间的路径等,进一步提升了分析的深度和广度。

  3. 动态性:能够实时反映系统或网络的变化,及时展现新增元素和调整后的结构关系,对于监控和管理系统状态尤为重要。

  4. 灵活性:用户可以根据需要选择不同的布局算法,调整图形的展示方式,更好地适应不同的分析场景。

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

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

相关文章

【系统架构设计师】计算机组成与体系结构 ③ ( 层次化存储结构 | 寄存器 | 高速缓存 | 内存 | 外存 )

文章目录 一、层次化存储结构1、层次化存储结构2、层次化存储结构 - 示例说明3、程序员可操作的部分 计算机 采用 分级存储结构 , 主要目的是 为了 解决 容量 / 价格 / 速度 之间的矛盾 ; 一、层次化存储结构 1、层次化存储结构 计算机 存储器 按照存储速度 由快到慢 进行排序 …

吐血推荐!3款视频生成工具,全部国产,都免费

AI视频大模型的爆发,让创作爆款视频不再是专业人士的能力。 今天二师兄给大家推荐3款免费的视频生成工具。 01 可灵 推荐指数 : 五颗星 先看效果 可灵大模型测试 可灵大模型是快手AI团队自主研发的视频生成大模型,具备强大的视频创作能力&a…

教程:在 Kubernetes 集群上部署 WordPress 网站

WordPress 是专为每个人设计的开源软件,强调创建网站、博客或应用程序的可访问性、性能、安全性和易用性。WordPress 是一个基于 PHP 的内容管理系统(CMS),使用 MySQL 作为数据存储,目前很多网站、电商独立站、个人博客…

推荐系统三十六式学习笔记:原理篇.模型融合14|一网打尽协同过滤、矩阵分解和线性模型

目录 从特征组合说起FM模型1.原理2.模型训练3.预测阶段4.一网打尽其他模型5.FFM 总结 在上一篇文章中,我们讲到了使用逻辑回归和梯度提升决策树组合的模型融合办法,用于CTR预估,给这个组合起了个名字,叫“辑度组合”。这对组合中&…

个人支付系统实现

基础首页: 订单: 智能售卡系统 基于webmanworkerman开发 禁用函数检查 使用这个脚本检查是否有禁用函数。命令行运行curl -Ss https://www.workerman.net/check | php 如果有提示Function 函数名 may be disabled. Please check disable_functions in …

线程池FutureTask浅谈

一,概述 FuturnTask实现了Future与Runnable接口,笔者知道,ThreadPoolExecutor#submit可以传入Callable接口而非Runnable,区别点在于Callable可以返回值,而整个FuturnTask可以理解为Callable设计,用来优雅地异步获取执行结果,无需手动Condition去实现。 围绕此,需知道…

鸿蒙开发系统基础能力:【@ohos.wallpaper (壁纸)】

壁纸 说明: 本模块首批接口从API version 7开始支持。后续版本的新增接口,采用上角标单独标记接口的起始版本。 导入模块 import wallpaper from ohos.wallpaper;WallpaperType 定义壁纸类型。 系统能力: 以下各项对应的系统能力均为SystemCapability…

【项目实训】falsk后端连接数据库以及与前端vue进行通信

falsk连接数据库 我们整个项目采用vueflaskmysql的框架,之前已经搭建好了mysql数据库,现在要做的是使用flask连接到数据库并测试 安装flask 首先安装flask pip install flask 进行数据库连接 数据库连接需要使用到pymysql库以及flask库 连接数据库…

Linux-引导过程与服务控制

目录 一、Linux操作系统引导过程 1、引导过程总览 2、引导过程详解 2.1、开机自检(BIOS) 2.2、 MBR引导 2.3、GRUB菜单 2.4、加载内核(kernel) 2.5、init进程初始化 3、系统初始化进程 3.1、Systemd单元类型 3.2、运行级别所对应的 Systemd 目…

SherlockChain:基于高级AI实现的智能合约安全分析框架

关于SherlockChain SherlockChain是一款功能强大的智能合约安全分析框架,该工具整合了Slither工具(一款针对智能合约的安全工具)的功能,并引入了高级人工智能模型,旨在辅助广大研究人员针对Solidity、Vyper和Plutus智…

前端 Array.sort() 源码学习

源码地址 V8源码Array 710行开始为sort()相关 Array.sort()方法是那种排序呢&#xff1f; 去看源码主要是源于这个问题 // In-place QuickSort algorithm. // For short (length < 22) arrays, insertion sort is used for efficiency.源码中的第一句话就回答了我的问题…

Potato(土豆)一款轻量级的开源文本标注工具(二)

示例项目&#xff08;模版&#xff09; Potato 旨在提高数据标注的可复制性&#xff0c;并降低研究人员设置新标注任务的成本。因此&#xff0c;Potato 提供了一系列预定义的示例项目&#xff0c;并欢迎公众向项目中心贡献。如果您使用 Potato 进行了自己的标注工作&#xff0…

海思平台使用ITTP_Stream调试sensor

目录 相关资料1.ISP相关资料2.MIPI RX相关资料3.sensor资料4.MIPI标准 准备工作1.准备sensor驱动2.准备sample vio3.准备上位机和下位机程序 运行1.只运行HiPQTool1.1.板端运行1.2.PC端运行HiPQTool 2.使用ITTP_Stream2.1.板端运行2.2.打开上位机软件 相关资料 1.ISP相关资料 …

uniapp开发手机APP、H5网页、微信小程序、长列表插件

ml-list 插件地址&#xff1a;https://ext.dcloud.net.cn/plugin?id18928 ml-list介绍 1、ml-list 列表组件&#xff0c;包含基本列表样式、可扩展插槽机制、长列表性能优化、多端兼容。 2、ml-list 低代码列表&#xff0c;帮助使用者快速构建列表&#xff0c;简单配置&…

秋招突击——6/26~6/27——复习{二维背包问题——宠物小精灵之收服}——新作{串联所有单词的字串}

文章目录 引言复习二维背包问题——宠物小精灵之收服个人实现重大问题 滚动数组优化实现 新作串联所有单词的字串个人实现参考实现 总结 引言 今天应该是舟车劳顿的一天&#xff0c;头一次在机场刷题&#xff0c;不学习新的东西了&#xff0c;就复习一些之前学习的算法了。 复…

有哪些方法可以恢复ios15不小心删除的照片?

ios15怎么恢复删除的照片&#xff1f;在手机相册里意外删除了重要的照片&#xff1f;别担心&#xff01;本文将为你介绍如何在iOS 15系统中恢复已删除的照片。无需专业知识&#xff0c;只需要按照以下步骤操作&#xff0c;你就能轻松找回宝贵的回忆。 一、从iCloud云端恢复删除…

SRC公益上分的小技巧一

前言 之前发布的文章&#xff0c;例如SRC中的一些信息收集姿势- Track 知识社区 - 掌控安全在线教育 - Powered by 掌控者 里面就有提到若依系统&#xff0c;默认账号密码非常简单 是 admin / admin123 但是&#xff0c;往往我们去挖掘的时候很容易出现 这说明了若依系统的门…

【Linux:文件描述符】

文件描述符&#xff1a; 文件描述符的分配原则&#xff1a;最小未分配原则 每一个进程中有一个task_struct结构体&#xff08;PCB)&#xff0c;而task_struct中含有struct file_sturct*file的指针&#xff0c;该指针指向了一个struct files_struct的结构体该结构体中含有一个f…

三大工作流引擎技术Activiti、Flowable、Camunda选型指南

文章目录 前言1 流程引擎发展历程2 流程引擎主要概念BPM (Business Process Management)BPMN (Business Process Model and Notation)CMMN (Case Management Model and Notation)DMN (Decision Model and Notation)事件&#xff08;Event&#xff09;顺序流&#xff08;Sequenc…

用一个实例看如何分享大量照片 续篇二,关于Exif (Exchangeable Image File) - 可交换图像文件

续篇二&#xff1a;说说关于照片隐含的 Exif (Exchangeable Image File) 可交换图像文件 数码照片的Exif 参数有很多&#xff0c;重要的Exif信息&#xff1a;拍摄日期、时间、拍摄器材、GPS信息。 当然这主要对自己的档案有意义&#xff0c;如果放到网上还是建议抹去这些信息。…