当前位置: 首页 > news >正文

【图的应用二:最短路径】- 用 C 语言实现迪杰斯特拉算法和弗洛伊德算法

目录

一、最短路径

二、迪杰斯特拉算法

三、弗洛伊德算法


 


一、最短路径

假若要在计算机上建立一个交通咨询系统,则可以采用图的结构来表示实际的交通网络。如下图所示,图中顶点表示城市,边表示城市间的交通联系。

这个咨询系统可以回答旅客提出的各种问题。例如,一位旅客要从 A 城到 B 城,他希望选择一条中转次数最少的路线。假设图中每一站都需要换车,则这个问题反映到图上就是找一条顶点 A 到顶点 B 所包含边的数目最少的路径。只需从顶点 A 出发对图做广度优先搜索,一旦遇到顶点 B 就终止,由此所得的广度优先生成树上,从根顶点 A 到顶点 B 的路径就是中转次数最少的路径。

但是,这只是一类最简单的图的最短路径问题。有时,对于旅客来说,可能更关心的是节省交通费用;而对于司机来说,里程和速度则是他们感兴趣的信息。为了在图上表示有关信息,可对边赋予权,权的值表示两城市间的距离,或途中所需时间,或交通费用等。此时路径长度的度量就不再是路径上边的数目,而是路径上边的权值之和。考虑到交通图的有向性,例如,汽车的上山和下山,轮船的顺水和逆水,所花费的时间或代价就不相同,所以交通网往往是用带权有向网表示。在带权有向网中,习惯上称路径上的第一个顶点为源点(Source),最后一个顶点为终点(Destination)

下面主要讨论两种最常见的最短路径问题:一种是求从某个源点到其余各顶点的最短路径,另一种是求每一对顶点之间的最短路径


二、迪杰斯特拉算法

单源点的最短路径问题:给定带权有向图 G 和源点 v0,求从 v0 到 G 中其余各顶点的最短路径。

迪杰斯特拉(Dijkstra)提出了一个按路径长度递增的次序产生最短路径的算法,称为迪杰斯特拉算法。

(1) 迪杰斯特拉算法的求解过程

对于网 N = (V, E),将 N 中的顶点分为两组:

第一组 S:已求出的最短路径的终点集合(初始时只包含源点 v0)。

第二组 V - S:尚未求出的最短路径的顶点集合(初始时为 V - {v0})。

算法将按各顶点与 v0 间最短路径长度递增的次序,逐个将集合 V - S 中的顶点加入到集合 S 中去。在这个过程中,总保持从 v0 到集合 S 中各顶点的路径长度始终不大于到集合 V - S 中各顶点的路径长度。

这种求解方法能确保是正确的,因为假设 S 为已求得最短路径的终点的集合,则可证明:下一条最短路径(设其终点为 x)或是边 (v0, x),或是中间只是经过 S 中的顶点而最后到达顶点 x 的路径

这可用反证法来证明。假设此路径上有一个顶点不在 S 中,则说明存在一条终点不在 S 而长度比此路径短的路径。但是,这是不可能的,因为算法是按路径长度递增的次序来产生最短路径的,故长度比此路径短的所有路径均已产生,它们的终点必定在 S 中,即假设不成立。

(2) 迪杰斯特拉算法的实现

假设用带权的邻接矩阵 arcs 来表示带权有向网 G

算法的实现要引入以下辅助的数据结构:

  1. 一维数组 S[i]:记录从源点 v0 到终点 vi 是否已被确定最短路径长度,true 表示确定,false 表示尚未确定。

  2. 一维数组 Path[i]:记录从源点 v0 到终点 vi 的当前最短路径上 vi 的直接前驱顶点序号。其初值为:如果从 v0 到 vi 有弧,则 Path[i] 为 v0;否则为 -1

  3. 一维数组 D[i]:记录从源点 v0 到终点 vi 的当前最短路径长度。其初值为:如果从 v0 到 vi 有弧,则 D[i] 为弧上的权值;否则为 \infty

    最短路径为 D[k] = Min{ D[i] | },求得从源点到 vk 的最短路径后,将 vk 加入到第一组顶点集 S 中

    每当加入一个新的顶点到顶点集 S,对第二组剩余的各个顶点而言,多了一个 "中转" 顶点,从而多了一个 "中转" 路径,所以要对第二组剩余的各个顶点的最短路径长度进行更新

    原来从 v0 到 vi 的最短路径长度为 D[i],加入 vk 之和,以 vk 作为中间顶点的 "中转" 路径长度为:D[k] + G.arcs[k][i],若 D[k] + G.arcs[k][i] < D[i],则用 D[k] + G.arcs[k][i] < D[i] 取代 D[i]

AMGraph.h

#pragma once
​
typedef char VertexType;
typedef int ArcType;
​
#define DEFAULT_CAPACITY 2
​
typedef struct AMGraph
{VertexType* vertices;ArcType** arcs;int vSize;int aSize;int capacity;
}AMGraph;
​
void AMGraphInit(AMGraph* pg);
​
void ShowAdjMatrix(AMGraph* pg);
​
int GetVertexPos(AMGraph* pg, VertexType v);
​
void InsertVertex(AMGraph* pg, VertexType v);
void InsertArc(AMGraph* pg, VertexType v1, VertexType v2, ArcType cost);
​
// 迪杰斯特拉算法
void ShortestPath_DIJ(AMGraph* pg, VertexType v, int* D, int* Path);

AMGraph.c

#include "AMGraph.h"
#include <assert.h>
#include <stdlib.h>
#include <stdio.h>
#include <stdbool.h>
​
void AMGraphInit(AMGraph* pg)
{assert(pg);pg->vSize = pg->aSize = 0;pg->capacity = DEFAULT_CAPACITY;
​pg->vertices = (VertexType*)malloc(sizeof(VertexType) * pg->capacity);assert(pg->vertices);
​pg->arcs = (ArcType**)malloc(sizeof(ArcType*) * pg->capacity);assert(pg->arcs);for (int i = 0; i < pg->capacity; ++i){pg->arcs[i] = (ArcType*)malloc(sizeof(ArcType) * pg->capacity);assert(pg->arcs[i]);for (int j = 0; j < pg->capacity; ++j){if (i == j)pg->arcs[i][j] = 0;elsepg->arcs[i][j] = INT_MAX;}}
}
​
void ShowAdjMatrix(AMGraph* pg)
{assert(pg);printf("   ");  // 输出 3 个空格for (int i = 0; i < pg->vSize; ++i){printf("%c  ", pg->vertices[i]);}printf("\n");
​for (int i = 0; i < pg->vSize; ++i){printf("%c  ", pg->vertices[i]);for (int j = 0; j < pg->vSize; ++j){if (pg->arcs[i][j] == INT_MAX)printf("#  ");  // 用 # 代替 ∞elseprintf("%-3d", pg->arcs[i][j]);}printf("\n");}
}
​
int GetVertexPos(AMGraph* pg, VertexType v)
{assert(pg);for (int i = 0; i < pg->vSize; ++i){if (pg->vertices[i] == v)return i;}return -1;
}
​
void InsertVertex(AMGraph* pg, VertexType v)
{assert(pg);// 考虑是否需要扩容if (pg->vSize == pg->capacity){VertexType* tmp1 = (VertexType*)realloc(pg->vertices, sizeof(VertexType) * 2 * pg->capacity);assert(tmp1);pg->vertices = tmp1;
​ArcType** tmp2 = (ArcType**)realloc(pg->arcs, sizeof(ArcType*) * 2 * pg->capacity);assert(tmp2);pg->arcs = tmp2;for (int i = 0; i < pg->capacity; ++i){ArcType* tmp3 = (ArcType*)realloc(pg->arcs[i], sizeof(ArcType) * 2 * pg->capacity);assert(tmp3);pg->arcs[i] = tmp3;for (int j = pg->capacity; j < 2 * pg->capacity; ++j){pg->arcs[i][j] = INT_MAX;}}for (int i = pg->capacity; i < 2 * pg->capacity; ++i){pg->arcs[i] = (ArcType*)malloc(sizeof(ArcType) * 2 * pg->capacity);assert(pg->arcs[i]);for (int j = 0; j < 2 * pg->capacity; ++j){if (i == j)pg->arcs[i][j] = 0;elsepg->arcs[i][j] = INT_MAX;}}
​pg->capacity *= 2;}// 插入顶点pg->vertices[pg->vSize++] = v;
}
​
void InsertArc(AMGraph* pg, VertexType v1, VertexType v2, ArcType cost)
{assert(pg);int pos1 = GetVertexPos(pg, v1);int pos2 = GetVertexPos(pg, v2);if (pos1 == -1 || pos2 == -1)return;
​if (pg->arcs[pos1][pos2] != INT_MAX)return;
​pg->arcs[pos1][pos2] = cost;++pg->aSize;
}
​
// 迪杰斯特拉算法的实现
void ShortestPath_DIJ(AMGraph* pg, VertexType v, int* D, int* Path)
{assert(pg);int pos = GetVertexPos(pg, v);if (pos == -1)return;
​bool* S = (bool*)malloc(sizeof(bool) * pg->vSize);assert(S);for (int i = 0; i < pg->vSize; ++i){S[i] = false;D[i] = pg->arcs[pos][i];if (i != pos && D[i] != INT_MAX)Path[i] = pos;elsePath[i] = -1;   }S[pos] = true;
​for (int i = 0; i < pg->vSize - 1; ++i){int min;int k;int flag = 1;for (int j = 0; j < pg->vSize; ++j){if (S[j] != false){continue;}if (flag){min = D[j];k = j;flag = 0;continue;}if (D[j] < min){min = D[j];k = j;}}
​S[k] = true;for (int j = 0; j < pg->vSize; ++j){
​if (S[j] == false && pg->arcs[k][j] != INT_MAX && D[k] + pg->arcs[k][j] < D[j]){D[j] = D[k] + pg->arcs[k][j];Path[j] = k;}}}
​free(S);
}

Test.c

#include "AMGraph.h"
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
​
int main()
{AMGraph g;AMGraphInit(&g);InsertVertex(&g, 'A');InsertVertex(&g, 'B');InsertVertex(&g, 'C');InsertVertex(&g, 'D');InsertVertex(&g, 'E');InsertVertex(&g, 'F');InsertArc(&g, 'A', 'C', 10);InsertArc(&g, 'A', 'E', 30);InsertArc(&g, 'A', 'F', 100);InsertArc(&g, 'B', 'C', 5);InsertArc(&g, 'C', 'D', 50);InsertArc(&g, 'D', 'F', 10);InsertArc(&g, 'E', 'D', 20);InsertArc(&g, 'E', 'F', 60);ShowAdjMatrix(&g);printf("\n");
​int* D = (int*)malloc(sizeof(int) * g.vSize);int* Path = (int*)malloc(sizeof(int) * g.vSize);assert(D && Path);ShortestPath_DIJ(&g, 'A', D, Path);
​for (int i = 1; i < g.vSize; ++i){if (D[i] == INT_MAX)printf("从 A 到 %c 没有路径!\n", g.vertices[i]);elseprintf("从 A 到 %c 的最短路径长度为:%d\n", g.vertices[i], D[i]);}free(D);free(Path);return 0;
}


三、弗洛伊德算法

求解每一对顶点之间的最短路径有两种方法:其一是分别以图中的每个顶点为源点共调用 n 次迪杰斯特拉算法;其二是采用下面介绍的弗洛伊德(Floyd)算法。两种算法的时间复杂度均为 O(n^3),但后者形式上较简单。

弗洛伊德算法仍然使用带权的邻接矩阵 arcs 来表示有向网 G,求从顶点 vi 到 vj 的最短路径。

算法的实现要引入以下辅助的数据结构:

  1. 二维数组 D[i][j]:记录顶点 vi 到 vj 之间的最短路径长度。

  2. 二维数组 Path[i][j]:最短路径上顶点 vj 的前一顶点的序号。

算法步骤:

将 vi 到 vj 的最短路径长度初始化,即 D[i][j] = G.arcs[i][j],然后进行 n 次比较和更新。

  1. 在 vi 和 vj 间加入顶点 v0,比较 (vi, vj) 和 (vi, v0, vj) 的路径长度,取其中较短者为 vi 到 vj 的中间顶点序号不大于 0 的最短路径。

  2. 在 vi 和 vj 间加入顶点 v1,得到 (vi, ..., v1) 和 (v1, ..., vj),其中 (vi, ..., v1) 是 vi 到 v1 的且中间顶点序号不大于 0 的最短路径,(v1, ..., vj) 是 v1 到 vj 的且中间顶点的序号不大于 0 的最短路径,这两条路径已在上一步中求出。比较 (vi, ...., v1, ..., vj) 与上一步求出的 vi 到 vj 的中间顶点序号不大于 0 的最短路径,取其中较短者作为 vi 到 vj 的中间顶点序号不大于 1 的最短路径。

  3. 依次类推,在 vi 和 vj 间加入顶点 vk,得到 (vi, ..., vk) 和 (vk, ..., vj),它们分别是从 vi 到 vk 和从 vk 到 vj 的中间顶点序号不大于 k - 1 的最短路径,将 (vi, ..., vk, ..., vj) 和已经得到的从 vi 到 vj 且中间顶点序号不大于 k - 1 的最短路径相比较,其长度较短者便是从 vi 到 vj 的中间顶点的序号不大于 k 的最短路径。这样,经过 n 次比较后,最后求得的必是从 vi 到 vj 的最短路径。按此方法,可用同时求得各对顶点间的最短路径。

void ShortestPath_Floyd(AMGraph* pg, int** D, int** Path)
{assert(pg);for (int i = 0; i < pg->vSize; ++i){for (int j = 0; j < pg->vSize; ++j){D[i][j] = pg->arcs[i][j];if (i != j && D[i][j] != INT_MAX)Path[i][j] = i;elsePath[i][j] = -1;}}
​for (int k = 0; k < pg->vSize; ++k){for (int i = 0; i < pg->vSize; ++i){for (int j = 0; j < pg->vSize; ++j){if (i != k && j != k && i != j){if (D[i][k] != INT_MAX && D[k][j] != INT_MAX &&D[i][k] + D[k][j] < D[i][j]){D[i][j] = D[i][k] + D[k][j];Path[i][j] = Path[k][j];}}}}}
}

Test.c

#include "AMGraph.h"
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
​
int main()
{AMGraph g;AMGraphInit(&g);InsertVertex(&g, 'A');InsertVertex(&g, 'B');InsertVertex(&g, 'C');InsertVertex(&g, 'D');InsertArc(&g, 'A', 'B', 1);InsertArc(&g, 'A', 'D', 4);InsertArc(&g, 'B', 'C', 9);InsertArc(&g, 'B', 'D', 2);InsertArc(&g, 'C', 'A', 3);InsertArc(&g, 'C', 'B', 5);InsertArc(&g, 'C', 'D', 8);InsertArc(&g, 'D', 'C', 6);ShowAdjMatrix(&g);printf("\n");
​int** D = (int**)malloc(sizeof(int*) * g.vSize);assert(D);for (int i = 0; i < g.vSize; ++i){D[i] = (int*)malloc(sizeof(int) * g.vSize);assert(D[i]);}int** Path = (int**)malloc(sizeof(int*) * g.vSize);assert(Path);for (int i = 0; i < g.vSize; ++i){Path[i] = (int*)malloc(sizeof(int) * g.vSize);assert(Path[i]);}ShortestPath_Floyd(&g, D, Path);
​for (int i = 0; i < g.vSize; ++i){for (int j = 0; j < g.vSize; ++j){printf("%d ", D[i][j]);}printf("\n");}printf("\n");for (int i = 0; i < g.vSize; ++i){for (int j = 0; j < g.vSize; ++j){printf("%d ", Path[i][j]);}printf("\n");}
​free(D);free(Path);return 0;
}

相关文章:

【图的应用二:最短路径】- 用 C 语言实现迪杰斯特拉算法和弗洛伊德算法

目录 一、最短路径 二、迪杰斯特拉算法 三、弗洛伊德算法 一、最短路径 假若要在计算机上建立一个交通咨询系统&#xff0c;则可以采用图的结构来表示实际的交通网络。如下图所示&#xff0c;图中顶点表示城市&#xff0c;边表示城市间的交通联系。 这个咨询系统可以回答旅…...

Qt之判断一个点是否在多边形内部(射线法)

算法思想: 以被测点Q为端点,向任意方向作射线(一般水平向右作射线),统计该射线与多边形的交点数。如果为奇数,Q在多边形内;如果为偶数,Q在多边形外。计数的时候会有一些特殊情况。这种方法适用于任意多边形,不需要考虑精度误差和多边形点给出的顺序,时间复杂度为O(n)…...

压力测试过程中内存溢出(堆溢出、栈溢出、持久代溢出)情况如何解决

在压力测试过程中&#xff0c;可能会遇到内存溢出的问题&#xff0c;其中常见的包括堆内存溢出、栈内存溢出和持久代溢出。解决这类问题需要首先理解各种内存溢出的原因和特点。 堆内存溢出&#xff1a;这种情况通常发生在稳定性压测一段时间后&#xff0c;系统报错&#xff0…...

【工业智能】音频信号相关场景

【工业智能】音频信号相关场景 DcaseDcase introduction&#xff1a;dcase2024有10个主题的任务&#xff1a; ASD硬件设备产品商 方法制造业应用场景 zenodo音频事件检测 与计算机视觉CV相对应&#xff0c;计算机听觉computer audition&#xff0c;简称CA。 Dcase 这里推荐一个…...

(PC+WAP)装修设计公司网站模板 家装公司网站源码下载

(PCWAP)装修设计公司网站模板 家装公司网站源码下载 PbootCMS内核开发的网站模板&#xff0c;该模板适用于装修设计、家装公司类等企业&#xff0c;当然其他行业也可以做&#xff0c;只需要把文字图片换成其他行业的即可&#xff1b; PCWAP&#xff0c;同一个后台&#xff0c…...

使用opencv实现图像中几何图形检测

1 几何图形检测介绍 1.1 轮廓(contours) 什么是轮廓&#xff0c;简单说轮廓就是一些列点相连组成形状、它们拥有同样的颜色、轮廓发现在图像的对象分析、对象检测等方面是非常有用的工具&#xff0c;在OpenCV 中使用轮廓发现相关函数时候要求输入图像是二值图像&#xff0c;这…...

补题与周总结:leetcode第 376 场周赛

文章目录 复盘与一周总结2967. 使数组成为等数数组的最小代价&#xff08;中位数贪心 回文数判断&#xff09;2968. 执行操作使频率分数最大&#xff08;中位数贪心 前缀和 滑窗&#xff09; 复盘与一周总结 wa穿了第3题&#xff0c;赛时其实想到了思路&#xff1a;中位数贪心…...

js指纹库,可跟踪用户唯一性

fingerprintjs官网 资料&#xff1a; Browserleaks - Check your browser for privacy leaks...

Shell三剑客:awk(内部变量)

一、$0 &#xff1a;完整的输入记录 [rootlocalhost ~]# awk -F: {print $0} passwd.txt root:x:0:0:root:/root:/bin/bash bin:x:1:1:bin:/bin:/sbin/nologin daemon:x:2:2:daemon:/sbin:/sbin/nologin adm:x:3:4:adm:/var/adm:/sbin/nologin lp:x:4:7:lp:/var/spool/lpd:/s…...

JVM中的虚拟机栈的动态链接部分存放到底是什么

在Java虚拟机&#xff08;JVM&#xff09;中&#xff0c;每个线程在执行一个方法时都会创建一个栈帧&#xff08;Stack Frame&#xff09;&#xff0c;栈帧中包含了方法的运行时数据。栈帧通常包括局部变量表、操作数栈、动态链接、方法返回地址等部分。 动态链接 动态链接&a…...

Leetcode 55 跳跃游戏

题意理解&#xff1a; 非负整数数组 nums, 最初位于数组的 第一个下标 。 数组中的每个元素代表你在该位置可以跳跃的最大长度。 需要跳到nums最后一个元素即为成功。 目标&#xff1a;是否能够跳到最后一个元素。 解题思路&#xff1a; 使用贪心算法来解题&#xff0c;需要理解…...

构建陪诊预约系统:技术实战指南

在医疗科技的飞速发展中&#xff0c;陪诊预约系统的应用为患者和陪诊人员提供了更为便捷和贴心的服务。本文将带领您通过技术实现&#xff0c;构建一个简单而实用的陪诊预约系统&#xff0c;以提升医疗服务的效率和用户体验。 技术栈选择 在开始之前&#xff0c;我们需要选择…...

windows和linux将文件删除至回收站【C++】【Go】语言实现

目录 C Windows平台 Linux平台 开平台&#xff0c;代码合并 Go 实现步骤 Go语言实现示例 go单独的windows版本实现 代码解释 C 在C中&#xff0c;将文件移动到回收站的实现在Linux和Windows平台上是不同的。首先&#xff0c;我会为你提供在Windows平台上实现的代码示例…...

10 Vue3中v-html指令的用法

概述 v-html主要是用来渲染富文本内容&#xff0c;比如评论信息&#xff0c;新闻信息&#xff0c;文章信息等。 v-html是一个特别不安全的指令&#xff0c;因为它会将文本以HTML的显示进行渲染&#xff0c;一旦文本里面包含一些恶意的js代码&#xff0c;可能会导致整个网页发…...

华为数通方向HCIP-DataCom H12-831题库(多选题:181-200)

第181题 如图所示,R1、R2、R3、R4都部署为SPF区域0,链路的cost值如图中标识。R1、R2R3、R4的Loopback0通告入OSPF。R1、R2、R3与R4使用Loopback0作为连接接口,建立BGP对等体关系,其中R4为RR设备,R1、R2、R3是R4的客户端。当R4的直连地址172.20,1,4/32通告入BGP后,以下关R…...

DC-磁盘管理

2023年全国网络系统管理赛项真题 模块B-Windows解析 题目 在DC2上安装及配置软RAID 5。在安装好的DC2虚拟机中添加三块10G虚拟磁盘。组成RAID 5,磁盘分区命名为卷标H盘:Raid5。手动测试破坏一块磁盘,做RAID磁盘修复,确认RAID 5配置完毕。配置步骤 关闭虚拟机,添加3块10G磁…...

使用Docker运行镜像文件与设置端口

1&#xff0c;创建镜像文件前准备 # 使用基础镜像FROM alpine:latest# 设置工作目录WORKDIR /app# 复制应用程序文件到镜像中COPY . .# 暴露容器的端口 不会自动将容器的端口映射到宿主机上 docker run -d -p <宿主机端口>:7080 <镜像名称>EXPOSE 7080# 定义容器启…...

Centos 8.5 Oracle12c安装

由于多次安装踩坑&#xff0c;所以本次写了一份12c安装的完整版。可以直接使用。 一、安装数据库基本信息 名称 值 主机名 database 操作系统 CentOS Linux release 8.5.2111 Oracle用户名/密码 oracle Oracle 版本 12c Enterprise Edition Release 12.2.0.1.0 oracle…...

Apache Tomcat httpoxy 安全漏洞 CVE-2016-5388 已亲自复现

Apache Tomcat httpoxy 安全漏洞 CVE-2016-5388 已亲自复现 漏洞名称漏洞描述影响版本 漏洞复现环境搭建漏洞利用 修复建议总结 漏洞名称 漏洞描述 在Apache Tomcat中发现了一个被归类为关键的漏洞&#xff0c;该漏洞在8.5.4(Application Server Soft ware)以下。受影响的是组…...

ChatGLM3-6B 的调用参数说明,chat 与stream_chat 接口函数的参数说明

ChatGLM3-6B 是一个语言大模型&#xff0c;最近在评估这个模型&#xff0c;但发现它的文档有限&#xff0c;只能从demo代码中猜测调用的参数的含义&#xff0c;准确度是有限的&#xff1b;于是&#xff0c;通过查看源代码来研究&#xff0c;目前整理笔记如下&#xff1a; Chat…...

idea大量爆红问题解决

问题描述 在学习和工作中&#xff0c;idea是程序员不可缺少的一个工具&#xff0c;但是突然在有些时候就会出现大量爆红的问题&#xff0c;发现无法跳转&#xff0c;无论是关机重启或者是替换root都无法解决 就是如上所展示的问题&#xff0c;但是程序依然可以启动。 问题解决…...

【JavaEE】-- HTTP

1. HTTP是什么&#xff1f; HTTP&#xff08;全称为"超文本传输协议"&#xff09;是一种应用非常广泛的应用层协议&#xff0c;HTTP是基于TCP协议的一种应用层协议。 应用层协议&#xff1a;是计算机网络协议栈中最高层的协议&#xff0c;它定义了运行在不同主机上…...

《Playwright:微软的自动化测试工具详解》

Playwright 简介:声明内容来自网络&#xff0c;将内容拼接整理出来的文档 Playwright 是微软开发的自动化测试工具&#xff0c;支持 Chrome、Firefox、Safari 等主流浏览器&#xff0c;提供多语言 API&#xff08;Python、JavaScript、Java、.NET&#xff09;。它的特点包括&a…...

电脑插入多块移动硬盘后经常出现卡顿和蓝屏

当电脑在插入多块移动硬盘后频繁出现卡顿和蓝屏问题时&#xff0c;可能涉及硬件资源冲突、驱动兼容性、供电不足或系统设置等多方面原因。以下是逐步排查和解决方案&#xff1a; 1. 检查电源供电问题 问题原因&#xff1a;多块移动硬盘同时运行可能导致USB接口供电不足&#x…...

vue3 字体颜色设置的多种方式

在Vue 3中设置字体颜色可以通过多种方式实现&#xff0c;这取决于你是想在组件内部直接设置&#xff0c;还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法&#xff1a; 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...

使用van-uploader 的UI组件,结合vue2如何实现图片上传组件的封装

以下是基于 vant-ui&#xff08;适配 Vue2 版本 &#xff09;实现截图中照片上传预览、删除功能&#xff0c;并封装成可复用组件的完整代码&#xff0c;包含样式和逻辑实现&#xff0c;可直接在 Vue2 项目中使用&#xff1a; 1. 封装的图片上传组件 ImageUploader.vue <te…...

鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院查看报告小程序

一、开发环境准备 ​​工具安装​​&#xff1a; 下载安装DevEco Studio 4.0&#xff08;支持HarmonyOS 5&#xff09;配置HarmonyOS SDK 5.0确保Node.js版本≥14 ​​项目初始化​​&#xff1a; ohpm init harmony/hospital-report-app 二、核心功能模块实现 1. 报告列表…...

ServerTrust 并非唯一

NSURLAuthenticationMethodServerTrust 只是 authenticationMethod 的冰山一角 要理解 NSURLAuthenticationMethodServerTrust, 首先要明白它只是 authenticationMethod 的选项之一, 并非唯一 1 先厘清概念 点说明authenticationMethodURLAuthenticationChallenge.protectionS…...

关于 WASM:1. WASM 基础原理

一、WASM 简介 1.1 WebAssembly 是什么&#xff1f; WebAssembly&#xff08;WASM&#xff09; 是一种能在现代浏览器中高效运行的二进制指令格式&#xff0c;它不是传统的编程语言&#xff0c;而是一种 低级字节码格式&#xff0c;可由高级语言&#xff08;如 C、C、Rust&am…...

【JavaWeb】Docker项目部署

引言 之前学习了Linux操作系统的常见命令&#xff0c;在Linux上安装软件&#xff0c;以及如何在Linux上部署一个单体项目&#xff0c;大多数同学都会有相同的感受&#xff0c;那就是麻烦。 核心体现在三点&#xff1a; 命令太多了&#xff0c;记不住 软件安装包名字复杂&…...