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

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


最短路径

当图是带权图时,把从一个顶点 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));//初始化所有顶点的直接前驱为-1Visited[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;//路径长度加1path[v] = u;//最短路径中,应是从u到vEnQueue(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.mzph.cn/news/871017.shtml

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

相关文章

centOS79中安装redis7.0

##red## &#x1f534; 大家好&#xff0c;我是雄雄&#xff0c;欢迎关注微信公众号&#xff0c;雄雄的小课堂。 一、前言 新服务器&#xff0c;一些环境是少不了要安装的&#xff0c;比如常见的redis&#xff0c;mysql&#xff0c;nginx等&#xff0c;今天&#xff0c;上次&a…

【Golang】map的使用

map声明的方式 //声明var m map[string]string//在使用map之前&#xff0c;先make&#xff0c;make的作用就是给map分配空间m make(map[string]string)m["lover"] "Yzx"m["friend1"] "Zxw"m["friend2"] "Zzc"…

C语言-gcc编译四步

gcc -E hello.c -o hello.i 预处理 gcc -S hello.i -o hello.s 编译 gcc -c hello.s -o hello.o 汇编 gcc hello.o -o hello.exe 链接

springboot增加过滤器后中文乱码

记录一下小问题 public class RepeatableHttpServletWrapper extends HttpServletRequestWrapper {private byte[] body;public RepeatableHttpServletWrapper(HttpServletRequest request) throws IOException {super(request);request.setCharacterEncoding("UTF-8&q…

泛微E-Cology WorkflowServiceXml SQL注入漏洞复现

0x01 产品简介 泛微e-cology是一款由泛微网络科技开发的协同管理平台,支持人力资源、财务、行政等多功能管理和移动办公。 0x02 漏洞概述 2024年7月,泛微官方发布了新补丁,修复了一处SQL注入漏洞。经分析,攻击者无需认证即可利用该漏洞,建议受影响的客户尽快修复漏洞。…

fortran快速排序算法,示例对一维数组进行排序

fortran快速排序算法&#xff0c;示例对一维数组进行排序 0. 引言1. 快速排序方法(QuickSqrt)代码实现2. 结语 0. 引言 快速排序&#xff08;QuickSort&#xff09;是一种常用的排序算法&#xff0c;采用分治策略实现。它的基本思想是通过一趟排序将待排序的数据分割成独立的两…

Lingo学习(一)——基本界面、解方程、变量

一、Lingo基本界面 【步骤】 1.双击打开Lingo 2.弹出一个对话框,点击Cancel左边的Never Register即可,其余内容用不到。 3:界面自动弹出名为“Lingo Model – Lingo 1”的窗口,用于书写代码。 4:以解方程的题目:x12为例,写完代码后,点击“红色的靶心”运行程序。 5:首先Lin…

第1章 初识 Express

1.1 什么是 Express Express 是一个简洁而灵活的 Node.js Web 应用框架&#xff0c;提供了一系列强大的特性用于开发 Web 和移动应用。它基于 Node.js 构建&#xff0c;并且与 Node.js 的非阻塞 I/O 模型无缝集成&#xff0c;使其非常适合于构建高性能的 Web 应用。 主要特点…

【办公软件】PPT使用轮子动画做圈动作

在实际的PPT制作中&#xff0c;我们可能会用到画圈的动作来强调重点。如下所示为最基础的画圈动作。 那么如何来做一个这样的动作呢&#xff1f; 首先在PPT中选择插入&#xff0c;选择形状椭圆 然后按Shift画图&#xff0c;即可画出一个正圆 然后使用绘图工具&#xff0c;将开关…

鸿蒙开发:Universal Keystore Kit(密钥管理服务)【密钥证明介绍及算法规格】

密钥证明介绍及算法规格 HUKS为密钥提供合法性证明能力&#xff0c;主要应用于非对称密钥的公钥的证明。 基于PKI证书链技术&#xff0c;HUKS可以为存储在HUKS中的非对称密钥对的公钥签发证书&#xff0c;证明其公钥的合法性。业务可以通过系统提供的根CA证书&#xff0c;逐级…

Photoshop套索工具使用指南:解锁自由选区的艺术

在Adobe Photoshop的强大工具箱中&#xff0c;套索工具组是每位图像处理爱好者与专业人士的得力助手。这组工具&#xff0c;包括套索工具、多边形套索工具和磁性套索工具&#xff0c;为用户提供了高度灵活的选择区域方式&#xff0c;无论是处理复杂的图像边缘还是进行精细的抠图…

VMware安装Ubuntu以及利用vscode远程Ubuntu

一、VMware安装Ubuntu &#xff08;1&#xff09;VMware安装Ubuntu主要参考此文VMware虚拟机安装Ubuntu22.04图文教程&#xff08;超详细&#xff01;&#xff01;&#xff01;&#xff09;。 &#xff08;2&#xff09;VMware密钥参考此文24年VMware 17密钥(附下载链接&#…

Android Settings应用 PreferenceScreen 条目隐藏实现和简单分析

Android Settings应用 PreferenceScreen 条目隐藏实现和简单分析 文章目录 Android Settings应用 PreferenceScreen 条目隐藏实现和简单分析一、前言二、隐藏实现1、xml 文件中隐藏PreferenceScreen 的某个条目2、普通Preference条目的隐藏的Java代码实现3、SwitchPreference条…

栈(Stack)与队列(Queue,Deque)

前言&#xff1a; 栈与队列在数据结构中用法都相对比较简单&#xff0c;是数据结构中经常用到的两种。 1.栈&#xff08;Stack&#xff09; &#xff08;1&#xff09;特点&#xff1a; 先入后出&#xff0c;后入先出。栈的底层就是一个数组&#xff08;java原生库中&#x…

在PyQt中为自己开发的软件实现远程文件“一机一码”授权管理实例

在使用PyQt搞软件开发时&#xff0c;开发者往往想要给自己的软件添加一个授权机制&#xff0c;只有当客户提供了授权码并且开发者将授权码放在授权管理系统的时候&#xff0c;客户端才能正常启动。这几天小陶就在捣鼓这个事&#xff0c;发现确实是可行的。 如果没有进行授权&a…

Ajax从零到实战

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 非常期待和您一起在这个小…

VR头显如何低延迟播放8K的RTSP|RTMP流

技术背景 我们在做Unity平台RTSP、RTMP播放器的时候&#xff0c;有公司提出来这样的技术需求&#xff0c;希望在头显播放全景的8K RTSP|RTMP直播流&#xff0c;8K的数据&#xff0c;对头显和播放器&#xff0c;都提出了新的要求&#xff0c;我们从几个方面&#xff0c;探讨下V…

SpringAOP的坑

AOP中几种常见的通知类型及其基本作用&#xff1a; Before&#xff1a;前置通知&#xff0c;在目标方法执行之前执行。After&#xff1a;后置通知&#xff0c;无论方法执行结果如何&#xff08;包括异常&#xff09;&#xff0c;都会在目标方法执行之后执行。AfterReturning&a…

Elasticsearch 更新指定字段

Elasticsearch 更新指定字段 准备条件查询数据更新指定字段更新子级字段 准备条件 以下查询操作都基于索引crm_clue来操作&#xff0c;索引已经建过了&#xff0c;本文主要讲Elasticsearch更新指定字段语句&#xff0c;下面开始写更新语句执行更新啦&#xff01; 查询数据 查…

MacOS 通过Docker安装宝塔面板搭建PHP开发环境

1、docker拉取ubuntu系统 docker pull ubuntu2、运行容器 docker run -i -t -d --name bt -p 20:20 -p 21:21 -p 80:80 -p 443:443 -p 888:888 -p 8888:8888 -p 3306:3306 -p 6379:6379 --privilegedtrue -v /Users/oi/Sites:/www/wwwroot ubuntu-v 后的 /Users/oi/Sites 代表…