408数据结构-图的应用2-最短路径 自学知识点整理

news2024/12/25 13:50:32

前置知识:最小生成树,图的遍历


最短路径

当图是带权图时,把从一个顶点 v 0 v_0 v0到图中其余任意一个顶点 v i v_i vi的一条路径所经过边上的权值之和,定位为该路径的带权路径长度,把带权路径长度最短的那条路径(不唯一)称为最短路径
求解最短路径的算法通常都依赖于一种性质,即两点之间的最短路径也包含了路径上其他顶点之间的最短路径。带权有向图 G G G的最短路径问题一般可分为两类:一类是单源最短路径,顾名思义“单个源头”,即求从图中某一顶点到其他各顶点的最短路径,可通过经典的 D i j k s t r a Dijkstra Dijkstra(迪杰斯特拉)算法求解;另一类是求每对顶点间的最短路径,可通过 F l o y e d Floyed Floyed(弗洛伊德)算法来求解。

回顾:BFS算法求解单源最短路径

广度优先搜索查找最短路径只是对无权图而言的。

无权图可以视为一种特殊的带权图,只是每条边的权值为 1 1 1

若图 G = ( V , E ) G=(V,E) G=(V,E)为无权图,定义从顶点 u u u到顶点 v v v的最短路径 d ( u , v ) d(u,v) d(u,v) u u u v v v的任何路径中最少的边数;若从 u u u v v v没有通路,则 d ( u , v ) = ∞ d(u,v)=\infin d(u,v)=
使用 B F S BFS BFS,我们可以求解一个满足上述定义的“无权图的单源最短路径问题”,这是由广度优先搜索总是按照距离由近到远来遍历图中每个顶点的性质决定的。
首先是预处理:使用邻接表对图进行存储。

#define MaxVertexNum 810
#define INF 0x7fffffff
//INF就是无穷大

typedef struct ArcNode {
	int adjvex;
	struct ArcNode* nextarc;
}ArcNode;
typedef struct VNode {
	int data;
	ArcNode* firstarc;
}VNode, AdjList[MaxVertexNum];
typedef struct {
	AdjList vertices;//邻接表存储图
	int vexnum, arcnum;
}ALGraph;
bool Visited[MaxVertexNum];//辅助数组标记顶点是否被访问

typedef struct LinkNode {
	int Vertex;//图的顶点编号
	struct LinkNode* next;
}LinkNode;
typedef struct {
	LinkNode* front, * rear;//链队列
}LinkQueue;

void InitQueue(LinkQueue& Q) {
	LinkNode* p = (LinkNode*)malloc(sizeof(LinkNode*));
	p->Vertex = -1;
	p->next = NULL;
	Q.front = Q.rear = p;//初始化队列
	return;
}

bool Q_Is_Empty(LinkQueue Q) {//队列判空
	return Q.front == Q.rear ? true : false;
}

void EnQueue(LinkQueue& Q, int i) {//新元素入队
	LinkNode* p = (LinkNode*)malloc(sizeof(LinkNode*));
	p->Vertex = i;
	p->next = NULL;
	Q.rear->next = p;
	Q.rear = p;
	return;
}

void DeQueue(LinkQueue& Q, int& x) {//队头元素出队
	LinkNode* p = Q.front->next;
	x = p->Vertex;
	Q.front->next = p->next;
	if (Q.rear == p)Q.rear = Q.front;
	free(p);
	return;
}

B F S BFS BFS求解单源最短路径问题:即求顶点 u u u到其他顶点的最短路径。

//求顶点u到其他顶点的最短路径
void BFS_MIN_Distance(ALGraph G, LinkQueue& Q, int u) {
	int d[MaxVertexNum], path[MaxVertexNum], v;
	//d[i]表示从u到i结点的最短路径;path[i]表示顶点u到顶点i的最短路径上,顶点i的直接前驱
	memset(d, INF, sizeof(d));//初始化路径长度为无穷大
	memset(path, -1, sizeof(path));//初始化所有顶点的直接前驱为-1
	Visited[u] = true, d[u] = 0;
	EnQueue(Q, u);
	while (!Q_Is_Empty(Q)) {//BFS主过程
		DeQueue(Q, u);//队头元素出队
		for (ArcNode* w = G.vertices[u].firstarc; w != NULL; w = w->nextarc) {
			//遍历u的所有邻接点
			v = w->adjvex;
			if (!Visited[v]) {//若未访问
				Visited[v] = true;//标记访问
				d[v] = d[u] + 1;//路径长度加1
				path[v] = u;//最短路径中,应是从u到v
				EnQueue(Q, v);//顶点w入队
			}
		}
	}
	return;
}

B F S BFS BFS的完整代码可以看我的Github:传送门

其中,数组 d [   ] d[\ ] d[ ]是用来记录每一个顶点到原始顶点的最短路径的长度,而数组 p a t h [   ] path[\ ] path[ ]则是用来记录每一个顶点在这个最短路径上的直接前驱。通过 p a t h [   ] path[\ ] path[ ]数组,可以从某一个顶点出发,一直逆向找到原始顶点,从而确定最短路径上经过的所有顶点。

知识点回顾:广度优先生成树
以原始顶点为根节点,对无权图构建一棵广度优先生成树,它的深度(高度)一定是最小的,各顶点所在的层数,也直接反映了原始顶点到该顶点的距离。

D i j k s t r a Dijkstra Dijkstra算法求解单源最短路径问题

D i j k s t r a Dijkstra Dijkstra算法通过设置一个集合 S S S记录已求得的最短路径的顶点。初始时把源点 v 0 v_0 v0放入 S S S,集合 S S S每并入一个新顶点 v i v_i vi,都要修改源点到集合 V − S V-S VS中顶点当前的最短路径长度值。
构造过程中,设置三个辅助数组:

  • f i n a l [   ] final[\ ] final[ ]:标记各顶点是否已找到最短路径,即是否已归入集合 S S S
  • d i s t [   ] dist[\ ] dist[ ]:记录从源点 v 0 v_0 v0到其他各顶点当前的最短路径长度,它的初始值为:若从 v 0 v_0 v0 v i v_i vi有弧,则 d i s t [ i ] dist[i] dist[i]为弧上的权值;否则置 d i s t [ i ] dist[i] dist[i] ∞ \infin
  • p a t h [   ] path[\ ] path[ ] p a t h [ i ] path[i] path[i]表示从源点到顶点 i i i之间的最短路径的前驱结点。在算法结束时,可根据其值追溯得到源点 v 0 v_0 v0到顶点 v i v_i vi的最短路径。

假设从顶点 0 0 0出发,即 v 0 = 0 v_0=0 v0=0,集合 S S S最初只包含顶点 0 0 0,邻接矩阵 a r c s arcs arcs表示带权有向图, a r c s [ i ] [ j ] arcs[i][j] arcs[i][j]表示有向边 < i , j > <i,j> <i,j>的权值,若不存在有向边 < i , j > <i,j> <i,j>,则 a r c s [ i ] [ j ] arcs[i][j] arcs[i][j] ∞ \infin

D i j k s t r a Dijkstra Dijkstra算法的步骤如下(不考虑对 p a t h [   ] path[\ ] path[ ]的操作):

  1. 初始化:集合 S S S初始为 0 {0} 0 d i s t [   ] dist[\ ] dist[ ]初始值 d i s t [ i ] = a r c s [ 0 ] [ i ] ,   i = 1 , 2 , ⋯   , n − 1 dist[i]=arcs[0][i],\ i=1,2,\cdots ,n-1 dist[i]=arcs[0][i], i=1,2,,n1
  2. 从顶点集合 V − S V-S VS中选出 v j v_j vj,满足 d i s t [ j ] = M i n { d i s t [ i ] ∣ v i ∈ V − S } dist\left[ j \right]=Min\left\{ dist\left[ i \right]\left| {{v}_{i}}\in V-S \right. \right\} dist[j]=Min{dist[i]viVS} v j v_j vj就是当前求得的一条从 v 0 v_0 v0出发的最短路径的终点,令 S = S ⋃ { j } S=S\bigcup \left\{ j \right\} S=S{j}
  3. 修改从 v 0 v_0 v0出发到集合 V − S V-S VS上任意一个顶点 v k v_k vk可达的最短路径长度:若 d i s t [ j ] + a r c s [ j ] [ k ] < d i s t [ k ] dist\left[ j \right]+arcs\left[ j \right]\left[ k \right]<dist\left[ k \right] dist[j]+arcs[j][k]<dist[k],则更新 d i s t [ k ] = d i s t [ j ] + a r c s [ j ] [ k ] dist\left[ k \right]=dist\left[ j \right]+arcs\left[ j \right]\left[ k \right] dist[k]=dist[j]+arcs[j][k]

下面是王道书上的一个关于实例的讲解,读者可以自己耐心看看理解一下 D i j k s t r a Dijkstra Dijkstra算法的执行过程。
(图片来自王道考研408数据结构2025)
图片来自王道考研408数据结构2025图片来自王道考研408数据结构2025
显然, D i j k s t r a Dijkstra Dijkstra算法也是基于贪心策略的。
使用邻接矩阵表示时,时间复杂度为 O ( ∣ V ∣ 2 ) O(|V|^2) O(V2)。使用带权的邻接表表示时,虽然修改 d i s t [   ] dist[\ ] dist[ ]的时间可以减少,但由于在 d i s t [   ] dist[\ ] dist[ ]中选择最小分量的时间不变,所以时间复杂度仍为 O ( ∣ V ∣ 2 ) O(|V|^2) O(V2)
即使只求解从源点到特定顶点的最短路径,时间复杂度依旧为 O ( ∣ V ∣ 2 ) O(|V|^2) O(V2)

注意:边上带有负权值时, D i j k s t r a Dijkstra Dijkstra算法将失效。既有可能出现有更短的路径而无法修改的情况,也有可能出现负环而导致死循环的情况。
(对前一种情况,例如: v 0 − > v 1 = 10 ,   v 0 − > v 2 = 7 ,   v 1 − > v 2 = − 5 v_0->v_1=10,\ v_0->v_2=7,\ v_1->v_2=-5 v0>v1=10, v0>v2=7, v1>v2=5 D i j k s t r a Dijkstra Dijkstra算法会将 d i s t [ 2 ] dist[2] dist[2]置为 7 7 7而后无法更新,实际上有一条距离更短的路径存在)
(对后一种情况,例如: v 0 − > v 1 = 1 ,   v 1 − > v 0 = − 2 ,   v 1 − > v 2 = 1 v_0->v_1=1,\ v_1->v_0=-2,\ v_1->v_2=1 v0>v1=1, v1>v0=2, v1>v2=1,此时 v 0 v_0 v0 v 1 v_1 v1之间构成一段负环,越走路径长度越小,会导致 D i j k s t r a Dijkstra Dijkstra算法陷入死循环)

思考: D i j k s t r a Dijkstra Dijkstra算法与 P r i m Prim Prim算法有何异同之处?

答: D i j k s t r a Dijkstra Dijkstra算法的流程、操作与 P r i m Prim Prim算法都很相似,核心思想都是基于贪心策略实现。区别在于:
①目的不同: D i j k s t r a Dijkstra Dijkstra算法的目的是构建单源点的最短路径树, P r i m Prim Prim算法的目的是构建最小生成树。
②算法思路略有不同: P r i m Prim Prim算法从一个点开始,每次选择权值最小的边,将其连接到已构建的生成树上,直至所有顶点都已加入;而 D i j k s t r a Dijkstra Dijkstra算法每次找出到源点距离最近且为归入集合的点,并把它归入集合,同时以这个点为基础更新从源点到其他所有顶点的距离。
③适用的图不同: P r i m Prim Prim算法只能用于带权无向图, D i j k s t r a Dijkstra Dijkstra算法可用于带权有向图或带权无向图(权值必须非负)。
时间复杂度不同:两个算法的时间复杂度之间有所不同。

F l o y e d Floyed Floyed算法求各顶点之间最短路径问题

求所有顶点之间的最短路径问题描述如下:已知一个各边权值均非负的带权有向图,对任意两个顶点 v i ≠ v j v_i≠v_j vi=vj,要求求出 v i v_i vi v j v_j vj之间的最短路径和最短路径长度。
F l o y e d Floyed Floyed算法的实现基于动态规划 d p dp dp),即把一个大问题分解为若干个性质相同,但规模更小的子问题,通过求解这些子问题的最优解进而求得大问题的最优解。类似于贪心,但是贪心求解的是局部最优解,可能会出现得不到全局最优解的情况,而动态规划使用了状态转移方程,能够较好地规避这个问题。关于动态规划的更多内容请参见:动态规划基础 - OI Wiki。
F l o y e d Floyed Floyed算法的基本思想是:递推产生一个 n n n阶方阵序列 A ( − 1 ) , A ( 0 ) , ⋯   , A ( k ) , ⋯   , A ( n − 1 ) {{A}^{\left( -1 \right)}},{{A}^{\left( 0 \right)}},\cdots ,{{A}^{\left( k \right)}},\cdots ,{{A}^{\left( n-1 \right)}} A(1),A(0),,A(k),,A(n1),其中 A ( k ) [ i ] [ j ] {A}^{(k)}[i][j] A(k)[i][j]表示从顶点 v i v_i vi到顶点 v j v_j vj之间的路径长度, k k k表示绕行第 k k k个顶点的运算步骤(即增加第 k k k个顶点作为中继点, k = − 1 k=-1 k=1时表示无任何中继点)。初始时,对于任意两个顶点 v i v_i vi v j v_j vj,若它们之间存在边,则以此边上的权值作为它们之间的最短路径,若不存在则置为 ∞ \infin 。之后逐步尝试在原路径中加入顶点 k k k k = 0 , 1 , ⋯   , n − 1 k=0,1,\cdots ,n-1 k=0,1,,n1)作为中继顶点。若增加中继顶点后,得到的新路径比原来的路径长度要小,则以此新路径代替原路径。算法描述如下:

  • 定义一个 n n n阶方阵序列 A ( − 1 ) , A ( 0 ) , ⋯   , A ( k ) , ⋯   , A ( n − 1 ) {{A}^{\left( -1 \right)}},{{A}^{\left( 0 \right)}},\cdots ,{{A}^{\left( k \right)}},\cdots ,{{A}^{\left( n-1 \right)}} A(1),A(0),,A(k),,A(n1),其中,
  • A ( − 1 ) [ i ] [ j ] = a r c s [ i ] [ j ] {{A}^{\left( -1 \right)}}\left[ i \right]\left[ j \right]=arcs\left[ i \right]\left[ j \right] A(1)[i][j]=arcs[i][j]
  • A ( k ) [ i ] [ j ] = M i n { A ( k − 1 ) [ i ] [ j ] , A ( k − 1 ) [ i ] [ k ] + A ( k − 1 ) [ k ] [ j ] } , k = 0 , 1 , ⋯   , n − 1 {{A}^{\left( k \right)}}\left[ i \right]\left[ j \right]=Min\left\{ {{A}^{\left( k-1 \right)}}\left[ i \right]\left[ j \right],{{A}^{\left( k-1 \right)}}\left[ i \right]\left[ k \right]+{{A}^{\left( k-1 \right)}}\left[ k \right]\left[ j \right] \right\},k=0,1,\cdots ,n-1 A(k)[i][j]=Min{A(k1)[i][j],A(k1)[i][k]+A(k1)[k][j]},k=0,1,,n1

式中, A ( 0 ) [ i ] [ j ] {{A}^{\left( 0 \right)}}\left[ i \right]\left[ j \right] A(0)[i][j]是从顶点 v i v_i vi到顶点 v j v_j vj、中继顶点是 v 0 v_0 v0的最短路径的长度。 A ( k ) [ i ] [ j ] {{A}^{\left( k \right)}}\left[ i \right]\left[ j \right] A(k)[i][j]是从顶点 v i v_i vi v j v_j vj、中继顶点是序号不大于 k k k的顶点的最短路径的长度。 F l o y e d Floyed Floyed算法是一个迭代的过程,每迭代一次,在从 v i v_i vi v j v_j vj的最短路径上就多考虑了一个顶点;经过 n n n次迭代后,所得到的 A ( n − 1 ) [ i ] [ j ] {{A}^{\left( n-1 \right)}}\left[ i \right]\left[ j \right] A(n1)[i][j]就是 v i v_i vi v j v_j vj的最短路径长度,即方阵 A ( n − 1 ) {{A}^{\left( n-1 \right)}} A(n1)中就保存了任意一对顶点之间的最短路径长度。
下面是王道书上的一个关于实例的讲解,读者可以自己耐心看看理解一下 F l o y e d Floyed Floyed算法的执行过程。当然,我个人认为王道的课在这一块讲得更好,比书上的东西更加清晰一些,感兴趣的读者可以去看看相关视频:王道计算机考研 数据结构 P67 6.4_4_最短路径问题_Floyed算法。
(图片来自王道考研408数据结构2025)
图片来自王道考研408数据结构2025
其核心代码实现,其实非常简单:

for (int k = 0; k < n; ++k)
	for (int i = 0; i < n; ++i)
		for (int j = 0; j < n; ++j)
			if (A[i][j] > A[i][k] + A[k][j])
				A[i][j] = A[i][k] + A[k][j], path[i][j] = k;

F l o y e d Floyed Floyed算法的时间复杂度为 O ( ∣ V ∣ 3 ) O(|V|^3) O(V3)。因为其代码结构简单紧凑,不包含其他复杂的数据结构,因此隐含的常数系数是很小的(不容易被卡常),对中等规模的输入来说,它仍然是相当有效的。
F l o y e d Floyed Floyed算法允许图中有带负权值的边,但不允许有负权回路(这样的图里可能不存在最短路)。
F l o y e d Floyed Floyed算法同样适用于带权无向图。
求各顶点之间最短路径问题也可以使用 D i j k s t r a Dijkstra Dijkstra算法,前提是图中没有负权值的边。依次将所有顶点都作为源点,分别运行一次 D i j k s t r a Dijkstra Dijkstra算法,时间复杂度同样为 O ( ∣ V ∣ 3 ) O(|V|^3) O(V3)。当然, D i j k s t r a Dijkstra Dijkstra算法可以使用 堆/优先队列 优化,使得时间复杂度从 O ( n 2 ) O(n^2) O(n2)降到 O ( n l o g n ) O(nlogn) O(nlogn) n = ∣ V ∣ n=|V| n=V),不过408考研初试并不需要掌握这么高级的东西。

相关例题:【洛谷P3371】【模板】单源最短路径(弱化版) 解题报告

  • 这篇博客是我对 D i j k s t r a Dijkstra Dijkstra算法的应用,感兴趣的读者可以看看。另外我也写了这题 F l o y e d Floyed Floyed算法的40分解法,完整代码可以参看我的Github:传送门
  • 核心代码如下:
for (int k = 1; k <= n; ++k)
	for (int i = 1; i <= n; ++i)
		for (int j = 1; j <= n; ++j)
			if (A[i][j] > A[i][k] + A[k][j])
				A[i][j] = A[i][k] + A[k][j], path[i][j] = k;

另外求解最短路径问题的算法还有 B e l l m a n − f o r d Bellman-ford Bellmanford算法, S P F A SPFA SPFA算法等,408考研初试也并不要求掌握这些。不过我以前打 O I OI OI的时候学过,所以提一嘴,实际上我以前学的也不好就是了。


408考研初试中,对这部分主要考察手动推算相关算法的执行过程,对代码的考查频率较低。
以上。

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

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

相关文章

借人工智能之手,编织美妙歌词篇章

在音乐的领域中&#xff0c;歌词宛如璀璨的明珠&#xff0c;为旋律增添了无尽的魅力和情感深度。然而&#xff0c;对于许多创作者来说&#xff0c;编织出美妙动人的歌词并非易事。但如今&#xff0c;随着科技的飞速发展&#xff0c;人工智能为我们带来了全新的创作可能。 “妙…

ant design form动态增减表单项Form.List如何进行动态校验规则

项目需求&#xff1a; 在使用ant design form动态增减表单项Form.List时&#xff0c;Form.List中有多组表单项&#xff0c;一组中的最后一个表单项的校验规则是动态的&#xff0c;该组为最后一组时&#xff0c;最后一个表单项是非必填项&#xff0c;其他时候为必填项。假设动态…

有奖竞猜!斗牛士军团与法兰西骑士的终极之战,谁将笑傲欧洲之巅?

痛快看球&#xff0c;畅玩游戏&#xff0c;AGON爱攻带你进入酣畅淋漓的足球世界&#xff01; 7月15日&#xff0c;绿茵赛场硝烟再起&#xff0c;两支身披荣光的王者之师&#xff0c;一路过关斩将&#xff0c;最终会师决赛。一场万众瞩目的巅峰对决即将拉开帷幕&#xff0c;究竟…

Midjourney v6.5 可能会在“7月底”发布,并改进了真实感和皮肤纹理

Midjourney v6.5即将发布&#xff0c;这一更新将大幅提升图像的真实感和皮肤纹理&#xff0c;为用户带来更逼真的视觉体验。首席执行官David Holz在电话会议中宣布&#xff0c;新版本将提高图像清晰度&#xff0c;特别是在手部和皮肤细节上&#xff0c;同时改进Web应用程序和个…

对红酒品质进行数据分析(python)

http://t.csdnimg.cn/UWg2S 数据来源于这篇博客&#xff0c;直接下载好csv文件。 这篇内容均在VScode的jupyter notebook上完成&#xff0c;操作可以看我的另一篇博客&#xff1a;http://t.csdnimg.cn/69sDJ 一、准备工作 1. 导入数据库 #功能是可以内嵌绘图&#xff0c;并…

替换:show-overflow-tooltip=“true“ ,使用插槽tooltip,达到内容可复制

原生的show-overflow-tooltip“true” 不能满足条件&#xff0c;使用插槽自定义编辑&#xff1b; 旧code <el-table-column prop"reason" label"原因" align"center" :show-overflow-tooltip"true" /> <el-table-column pro…

构建实时银行应用程序:英国金融机构 Nationwide 为何选择 MongoDB Atlas

Nationwide Building Society 超过135年的互助合作 Nationwide Building Society&#xff08;以下简称“Nationwide”&#xff09; 是一家英国金融服务提供商&#xff0c;拥有超过 1500 万名会员&#xff0c;是全球最大的建房互助会。 Nationwide 的故事可以追溯到 1884 年&am…

ArcGIS Pro、ChatGPT、Python、InVEST等多技术融合的水文、生态、气候变化等地学领域科研及项目综合能力提升

在当前科学技术飞速发展的背景下&#xff0c;综合科研能力的提升对于推动各个领域的创新和发展具有重要的意义。在当前竞争激烈的科研环境中&#xff0c;掌握先进的数据处理与分析技术、深入了解前沿的研究领域、有效利用智能工具进行科研工作&#xff0c;已成为科研人员脱颖而…

html5——CSS3_文本样式属性

目录 字体样式 字体类型 字体大小 字体风格 字体的粗细 文本样式 文本颜色 排版文本段落 文本修饰和垂直对齐 文本阴影 字体样式 字体类型 p{font-family:Verdana,"楷体";} body{font-family: Times,"Times New Roman", "楷体";} …

CF1473E Minimum Path 题解(最短路,分层图最短路,较重要的套路)

题目描述&#xff1a; 题目 分析&#xff1a; 题目是要让我们求从 1 1 1 出发&#xff0c;到 i i i 的路径的最小权值。其中路径的权值定义为 路径上所有的边权和 减去最大边权 加上最小边权。这里有一个很秒的转化&#xff1a;可以把一条路径的权值理解为 必须将路径上的任…

【面试八股总结】单例模式实现详解

一、基本概念 单例设计模式是⼀种确保⼀个类只有⼀个实例&#xff0c;并提供⼀个全局访问点来访问该实例的创建模式。 关键概念&#xff1a; 一个私有构造函数&#xff1a;确保只能单例类自己创建实例一个私有静态变量&#xff1a;确保只有一个实例&#xff0c;私有静态变量用…

IDEA自动把接口中的方法注解填充到实现类中,勾选Copy JavaDoc即可

1. 目的 有一个Image接口类&#xff0c;接口中有getUserById方法&#xff0c;方法上有注释&#xff0c;实现类ImageImpl实现Image中的方法时&#xff0c;自动把接口中方法的注释也给带下来 具体案例如下 2. 接口类 有一个getUserById方法&#xff0c;方法上面有注释 3. 实现…

Java常用的API_02(正则表达式、爬虫)

Java正则表达式 七、正则表达式7.1 格式7.1.1 字符类注意字符类示例代码1例2 7.1.2 预定义字符预定义字符示例代码例2 7.1.3 区别总结 7.2 使用Pattern和Matcher类与直接使用String类的matches方法的区别。&#xff08;1&#xff09; 使用Pattern和Matcher类示例代码 &#xff…

JVM 之对象的结构与创建

1.对象的创建 1.1类加载 当Java 虚拟机遇到一条字节码 new 指令时&#xff0c;首先将去检查这个指令的参数是否能在常量池中定位到 一个类的符号引用&#xff0c;并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有&#xff0c;那 必须先执行相应的类加载过…

昇思MindSpore学习总结十五 ——基于Mindspore 实现BERT对话情绪识别

1、环境配置 根据实际情况&#xff0c;选择合适版本。 %%capture captured_output # 实验环境已经预装了mindspore2.2.14&#xff0c;如需更换mindspore版本&#xff0c;可更改下面mindspore的版本号 !pip uninstall mindspore -y !pip install -i https://pypi.mirrors.ustc…

用node.js写一个简单的图书管理界面——功能:添加,删除,修改数据

涉及到的模块&#xff1a; var fs require(‘fs’)——内置模块 var ejs require(‘ejs’)——第三方模块 var mysql require(‘mysql’)——第三方模块 var express require(‘express’)——第三方模块 var bodyParser require(‘body-parser’)——第三方中间件 需要…

华为HCIP Datacom H12-821 卷38

1.多选题 下面关于 BGP中的公认属性的描述&#xff0c;正确的是 A、公认必遵属性是所有BGP路由器都识别&#xff0c;且必须存在于Updata消息中心 B、BGP必须识别所有公认属性 C、公认属性分为公认必遵和可选过渡两种 D、公认任意属性是所有BGP造由器都可以识别&#xff0c…

217.贪心算法:加油站(力扣)

代码解决 class Solution { public:int canCompleteCircuit(vector<int>& gas, vector<int>& cost) {int curtotol 0; // 当前累积油量int tatol 0; // 总的油量减去总的花费油量int start 0; // 起始加油站的索引// 遍历所有加油站for (int i 0; i &…

【Android面试八股文】你说ARouter采用APT技术,那么谈一下你对APT技术的理解,还有那些框架是采用APT技术呀?JavaPoet

一、谈一下你对APT技术的理解 1.1 对APT技术的理解 APT(Annotation Processing Tool)是一种在编译期间处理注解的技术,它允许开发者在编译时扫描和处理 Java 源代码中的注解信息,生成额外的源代码、资源文件或者其他文件。以下是对APT技术的一些理解和应用场景: 工作原理…

期货量化交易客户端开源教学第九节——新用户注册

一、新用户注册界面设计&#xff1a; 注册时采用手机号注册&#xff0c;客户端发送新号注册申请由后台做审核&#xff0c;后台审核通过后向注册的手机号发送注册成功的消息。注册过的手机号不能再二次注册。 界面验证代码 private{ Private declarations }FVerf: AnsiString; …