prim算法求解最小生成树

合集下载

的最小生成树算法Prim与Kruskal算法的比较

的最小生成树算法Prim与Kruskal算法的比较

的最小生成树算法Prim与Kruskal算法的比较Prim算法和Kruskal算法都是常用的最小生成树算法,它们可以在给定的加权连通图中找到连接所有节点的最小权重边集合。

然而,这两种算法在实现细节和时间复杂度上有所不同。

本文将对Prim算法和Kruskal算法进行比较,并讨论它们的优缺点以及适用场景。

一、Prim算法Prim算法是一种贪心算法,它从一个起始节点开始,逐步扩展最小生成树的边集合,直到包含所有节点为止。

具体步骤如下:1. 选取一个起始节点作为最小生成树的根节点。

2. 在最小生成树的边集合中寻找与当前树集合相连的最小权重边,并将这条边添加到最小生成树中。

3. 将新添加的节点加入到树集合中。

4. 重复步骤2和3,直到最小生成树包含所有节点为止。

Prim算法的时间复杂度为O(V^2),其中V是节点的个数。

这是因为在每轮迭代中,需要从树集合以外的节点中找到与树集合相连的最小权重边,而在最坏情况下,可能需要检查所有的边。

二、Kruskal算法Kruskal算法是一种基于边的贪心算法,它按照边的权重从小到大的顺序依次选择边,并判断是否加入最小生成树中。

具体步骤如下:1. 初始化一个空的最小生成树。

2. 将所有边按照权重从小到大进行排序。

3. 依次检查每条边,如果这条边连接了两个不同的树(即不会形成环),则将这条边加入到最小生成树中。

4. 重复步骤3,直到最小生成树包含所有节点为止。

Kruskal算法使用并查集数据结构来快速判断连通性,时间复杂度为O(ElogE),其中E是边的个数。

排序边的时间复杂度为O(ElogE),而对每条边进行判断和合并操作的时间复杂度为O(E)。

三、比较与总结1. 时间复杂度:Prim算法的时间复杂度为O(V^2),而Kruskal算法的时间复杂度为O(ElogE)。

因此,在边的数量较大的情况下,Kruskal 算法的效率优于Prim算法。

2. 空间复杂度:Prim算法需要维护一个大小为V的优先队列和一个大小为V的布尔数组,而Kruskal算法需要维护一个大小为V的并查集。

prim算法求最小生成树c代码

prim算法求最小生成树c代码

一、概述在图论中,最小生成树是指在一个连通图中生成一棵包含图中所有顶点且边权值之和最小的树。

prim算法是一种常用的求取最小生成树的算法之一,其基本思想是从一个起始顶点开始,逐步选择与当前树相邻的并且权值最小的边,直到包含了图中所有的顶点为止。

本文将介绍prim算法的原理以及给出相应的C代码实现。

二、prim算法原理1. 初始化选择任意一个顶点作为起始顶点,并将其标记为已访问。

设置一个集合V来存放已经访问的顶点,初始化为空集合。

2. 重复以下步骤直到V包含了所有顶点:(1)遍历集合V中的所有顶点,找到与之相邻且未被访问过的顶点中边权值最小的顶点,将其加入集合V中。

(2)将找到的顶点与其相邻的边加入最小生成树中。

3. 输出最小生成树当集合V包含了所有顶点之后,即可输出最小生成树。

三、prim算法C代码实现以下是prim算法的C代码实现:#include <stdio.h>#include <limits.h>#define V 5 // 顶点个数int minKey(int key[], bool mstSet[]) {int min = INT_MAX, min_index;for (int v = 0; v < V; v++) {if (!mstSet[v] key[v] < min) {min = key[v], min_index = v;}}return min_index;}void printMST(int parent[], int graph[V][V]) {printf("Edge \tWeight\n");for (int i = 1; i < V; i++) {printf("d - d \td \n", parent[i], i, graph[i][parent[i]]);}void primMST(int graph[V][V]) {int parent[V]; // 存放最小生成树的结点int key[V]; // 存放与最小生成树相邻的边的权值bool mstSet[V]; // 存放已访问的顶点for (int i = 0; i < V; i++) {key[i] = INT_MAX, mstSet[i] = false;}key[0] = 0;parent[0] = -1;for (int count = 0; count < V - 1; count++) {int u = minKey(key, mstSet);mstSet[u] = true;for (int v = 0; v < V; v++) {if (graph[u][v] !mstSet[v] graph[u][v] < key[v]) { parent[v] = u, key[v] = graph[u][v];}}}printMST(parent, graph); }int m本人n() {int graph[V][V] = {{0, 2, 0, 6, 0},{2, 0, 3, 8, 5},{0, 3, 0, 0, 7},{6, 8, 0, 0, 9},{0, 5, 7, 9, 0}};primMST(graph);return 0;}```四、代码说明1. minKey函数该函数用于在尚未加入最小生成树的结点中找到与最小生成树相邻的边中权值最小的结点。

Prim算法详细步骤

Prim算法详细步骤

Prim算法详细步骤Prim算法是一种用于解决最小生成树问题的贪心算法。

它通过选择边的方式逐步构建最小生成树,从而使得树中所有边的权重之和最小。

本文将详细介绍Prim算法的步骤。

1. 首先,我们需要确定一个起始点作为最小生成树的根节点。

这个起始点可以是图中的任意一个顶点,我们可以根据具体问题的需求来选择。

2. 接下来,我们需要定义一个集合T来存放最小生成树的边。

一开始,集合T是空的。

3. 然后,我们需要定义一个集合V来存放已经加入最小生成树的顶点。

一开始,集合V中只包含起始点。

4. 然后,我们需要找到一条从集合V中的顶点到集合V之外的顶点的边,且该边的权重最小。

我们将这条边称为"最小权重边"。

首先,我们可以将起始点连接到任意一个集合V之外的顶点的边加入集合T,并将该顶点加入集合V。

5. 继续执行步骤4,每次选择一条连接集合V与集合V之外顶点的最小权重边,并将该边加入集合T。

同时,将该边连接的顶点加入集合V。

6. 重复执行步骤5,直到最小生成树中包含了图中的所有顶点。

7. 最后,我们得到的集合T即为最小生成树的边。

通过以上步骤,Prim算法可以找到一个最小生成树。

下面我们通过一个具体的例子来演示Prim算法的过程。

假设我们有一个无向加权图,其中包含了6个顶点和9条边。

我们以A作为起始点开始Prim算法。

初始状态:集合V:{A}集合T:{}步骤4:起始点A连接到B的边为最小权重边,加入集合T,并将B加入集合V。

集合V:{A, B}集合T:{(A, B)}步骤5:集合V中的顶点A连接到D的边为最小权重边,加入集合T,并将D加入集合V。

集合V:{A, B, D}集合T:{(A, B), (A, D)}步骤5:集合V中的顶点D连接到F的边为最小权重边,加入集合T,并将F加入集合V。

集合V:{A, B, D, F}集合T:{(A, B), (A, D), (D, F)}步骤5:集合V中的顶点F连接到G的边为最小权重边,加入集合T,并将G加入集合V。

最小生成树的Prim算法以及Kruskal算法的证明

最小生成树的Prim算法以及Kruskal算法的证明

最⼩⽣成树的Prim算法以及Kruskal算法的证明Prime算法的思路:从任何⼀个顶点开始,将这个顶点作为最⼩⽣成树的⼦树,通过逐步为该⼦树添加边直到所有的顶点都在树中为⽌。

其中添加边的策略是每次选择外界到该⼦树的最短的边添加到树中(前提是⽆回路)。

Prime算法的正确性证明:引理1:对于连通图中的顶点vi,与它相连的所有边中的最短边⼀定是属于最⼩⽣成树的。

引理2:证明:假设最⼩⽣成树已经建成;(vi, vj)是连接到顶点vi的最短边,在最⼩⽣成树中取出vi,断开连接到vi的边,则⽣成树被拆分成1、顶点vi2、顶点vj所在的连通分量(单独⼀个顶点也看作⼀个独⽴的连通分量)3、其余若⼲个连通分量(个数⼤于等于0)三个部分现在要重建⽣成树,就要重新连接之前被断开的各边虽然不知道之前被断开的都是哪⼏条边,但是可以通过这样⼀个简单的策略来重建连接:将vi分别以最⼩的成本逐个连接到这若⼲个互相分离的连通分量;具体来说,就是要分别遍历顶点vi到某个连通分量中的所有顶点的连接,然后选择其中最短的边来连接vi和该连通分量;⽽要将vi连接到vj所在的连通分量,显然通过边(vi, vj)连接的成本最低,所以边(vi, vj)必然属于最⼩⽣成树(如果连接到vi的最短边不⽌⼀条,只要任意挑选其中的⼀条(vi, vj)即可,以上的证明对于这种情况同样适⽤)。

这样我们就为原来只有⼀个顶点vi的⼦树添加了⼀个新的顶点vj及新边(vi, vj);接下来只要将这棵新⼦树作为⼀个连通⼦图,并且⽤这个连通⼦图替换顶点vi重复以上的分析,迭代地为⼦树逐个地添加新顶点和新边即可。

Kruskal算法:通过从⼩到⼤遍历边集,每次尝试为最⼩⽣成树加⼊当前最短的边,加⼊成功的条件是该边不会在当前已构建的图中造成回路,当加⼊的边的数⽬达到n-1,遍历结束。

Kruskal算法的正确性证明:Kruskal算法每次为当前的图添加⼀条不会造成回路的新边,其本质是逐步地连接当前彼此分散的各个连通分量(单个顶点也算作⼀个连通分量),⽽连接的策略是每次只⽤最⼩的成本连接任意两个连通分量。

简述prim算法的过程

简述prim算法的过程

简述prim算法的过程Prim算法是一种用于求解最小生成树的贪心算法,它把一个连通图分割成多个子图,使得每个子图都是一棵最小生成树,且这些子树的联合就是原连通图的最小生成树。

Prim算法是由安德鲁普里姆(Andrew.Prim)在1957年提出的,用于求解最短路径(解决连通度和权重问题)。

Prim算法一般步骤如下:(1)初始化:从图中任选一顶点,作为第一个顶点加入到最小生成树中;(2)循环:从剩余的顶点中,找到最小代价的边,将这条边加入到最小生成树中,并将这条边的顶点也加入到最小生成树中;(3)重复:重复步骤2,直到最小生成树中包括全部的顶点;(4)停止:最小生成树已经构造完成,停止算法的执行。

Prim算法是贪心算法的一种,它每次在可选的边中搜索代价最小的边,加入到最小生成树中,直至所有的顶点都在最小生成树中。

它的一般步骤可表示为:(1)从最小生成树中所有的顶点中,找出一个确定的顶点 u,它的邻居还未加入最小生成树;(2)从u的邻居v中,找出一个代价最小的边(u,v),将这条边加到最小生成树中;(3)将顶点v加入到最小生成树中;(4)重复步骤1到3,直到最小生成树中包括所有的顶点;深入分析Prim算法,我们可以发现它是一种贪心策略,它在设计上采用了“最优化原则”,即每次都选择代价最小的边加入到最小生成树中,而不管这个边是否有利于求解最小生成树的问题,因而贪心算法的实现对“最优策略”的选择是关键。

Prim算法的时间复杂度取决于边的存储结构,如果存储为邻接表,其时间复杂度为O(V2),如果存储为邻接矩阵,其时间复杂度为O(V2+E)。

其中,V为顶点数,E为边数。

Prim算法在现实生活中有着广泛的应用,比如电路设计时,需要求最小生成树,以此达到最短路径、最小花费的目的;另外,它还可以用于网络路由的设计,最小化网络的延迟;此外,它还可以用于求解旅行商问题,最小化客户的费用等等。

总之,Prim算法是一种有效的求解最小生成树的方法,它通过不断地在可选边中寻找代价最小的边,来构建最小生成树;在现实生活中,它也有着广泛的应用。

PRIM算法求最小生成树

PRIM算法求最小生成树

xx学院《数据结构与算法》课程设计报告书课程设计题目 PRIM算法求最小生成树院系名称计算机科学与技术系专业(班级)姓名(学号)指导教师完成时间一、问题分析和任务定义在该部分中主要包括两个方面:问题分析和任务定义;1 问题分析本次课程设计是通过PRIM(普里姆)算法,实现通过任意给定网和起点,将该网所对应的所有生成树求解出来。

在实现该本设计功能之前,必须弄清以下三个问题:1.1 关于图、网的一些基本概念1.1.1 图图G由两个集合V和E组成,记为G=(V,E),其中V是顶点的有穷非空集合,E是V中顶点偶对的有穷集,这些顶点偶对称为边。

通常,V(G)和E(G)分别表示图G的顶点集合和边集合。

E(G)也可以为空集。

则图G只有顶点而没有边。

1.1.2 无向图对于一个图G,若边集E(G)为无向边的集合,则称该图为无向图。

1.1.3 子图设有两个图G=(V,E)G’=(V’,),若V’是V的子集,即V’⊆V ,且E’是E的子集,即E’⊆E,称G’是G的子图。

1.1.4 连通图若图G中任意两个顶点都连通,则称G为连通图。

1.1.5 权和网在一个图中,每条边可以标上具有某种含义的数值,该数值称为该边的权。

把边上带权的图称为网。

如图1所示。

1.2 理解生成树和最小生成树之间的区别和联系1.2.1 生成树在一个连通图G中,如果取它的全部顶点和一部分边构成一个子图G’,即:V(G’)= V(G)和E(G’)⊆E(G),若边集E(G’)中的边既将图中的所有顶点连通又不形成回路,则称子图G’是原图G的一棵生成树。

1.2.2 最小生成树图的生成树不是唯一的,把具有权最小的生成树称为图G的最小生成树,即生成树中每条边上的权值之和达到最小。

如图1所示。

图1.网转化为最小生成树1.3 理解PRIM(普里姆)算法的基本思想1.3.1 PRIM算法(普里姆算法)的基本思想假设G =(V,E)是一个具有n个顶点的连通网,T=(U,TE)是G的最小生成树,其中U是T的顶点集,TE是T的边集,U和TE的初值均为空集。

采用普里姆算法和克鲁斯卡尔算法,求最小生成树

采用普里姆算法和克鲁斯卡尔算法,求最小生成树

采用普里姆算法和克鲁斯卡尔算法,求最小生成树普利姆算法(Prim's Algorithm)和克鲁斯卡尔算法(Kruskal's Algorithm)是求解最小生成树的两种常用方法。

最小生成树是指连接图中所有节点,且边的权重和最小的树。

这两种算法各有特点,在不同的场景中使用。

1.普利姆算法:适用于边稠密的图普利姆算法是一种贪心算法,从一个节点开始,不断选择与当前树相连的、权重最小的边,并将该边连接的节点加入树中,直到所有节点都被遍历完。

这样就得到了最小生成树。

以下是普利姆算法的伪代码:1.创建一个空的树,用于保存最小生成树2.选择一个起始节点,将其加入树中3.从树中已有的节点出发,找到与树相连的边中权重最小的边4.将找到的边连接的节点加入树中5.重复步骤3和4,直到所有节点都加入树中普利姆算法的时间复杂度为O(ElogV),其中E为边的数量,V为节点的数量。

2.克鲁斯卡尔算法:适用于边稀疏的图克鲁斯卡尔算法是一种基于排序和并查集的贪心算法,按照边的权重从小到大的顺序选择,并判断是否会构成环。

如果不会构成环,则选择该边,并将其加入最小生成树中,直到所有节点都被连接。

以下是克鲁斯卡尔算法的伪代码:1.创建一个空的树,用于保存最小生成树2.将所有边按权重从小到大排序3.创建一个并查集,用于判断边是否会构成环4.遍历排序后的边,对于每条边,判断其连接的两个节点是否属于同一个集合(即是否会构成环)5.如果不会构成环,则选择该边,并将其加入树中,同时将该边连接的两个节点合并到同一个集合中6.重复步骤4和5,直到所有节点都连接在一起克鲁斯卡尔算法的时间复杂度为O(ElogE),其中E为边的数量。

这两种算法的应用场景有所不同。

如果要求解的图是边稠密的(即边的数量接近节点数量的平方),则使用普利姆算法更为高效。

因为普利姆算法的时间复杂度与边的数量有关,所以处理边稠密的图会更快一些。

而对于边稀疏的图(即边的数量接近节点数量的线性),克鲁斯卡尔算法更加适用,因为它的时间复杂度与边的数量有关。

普里姆算法(Prim)求最小生成树C程序

普里姆算法(Prim)求最小生成树C程序

程序测试用例如下:程序运行过程截图:源程序清单如下:#include <stdio.h>#define n 6#define MaxNum 10000 /*定义一个最大整数*/2 6 4 43 5 5 1 5 6 1 2 43 5 6/*定义邻接矩阵类型*/typedef int adjmatrix[n+1][n+1]; /*0号单元没用*/typedef struct{int fromvex,tovex;int weight;}Edge;typedef Edge *EdgeNode;int arcnum; /*边的个数*//*建立图的邻接矩阵*/void CreatMatrix(adjmatrix GA){int i,j,k,e;printf("图中有%d个顶点\n",n);for(i=1;i<=n;i++){for(j=1;j<=n;j++){if(i==j){GA[i][j]=0; /*对角线的值置为0*/}else{GA[i][j]=MaxNum; /*其它位置的值置初始化为一个最大整数*/ }}}printf("请输入边的个数:");scanf("%d",&arcnum);printf("请输入边的信息,按照起点,终点,权值的形式输入:\n");for(k=1;k<=arcnum;k++){scanf("%d,%d,%d",&i,&j,&e); /*读入边的信息*/GA[i][j]=e;GA[j][i]=e;}}/*初始化图的边集数组*/void InitEdge(EdgeNode GE,int m){int i;for(i=1;i<=m;i++){GE[i].weight=0;}}/*根据图的邻接矩阵生成图的边集数组*/void GetEdgeSet(adjmatrix GA,EdgeNode GE){ int i,j,k=1;for(i=1;i<=n;i++){for(j=i+1;j<=n;j++){if(GA[i][j]!=0&&GA[i][j]!=MaxNum){GE[k].fromvex=i;GE[k].tovex=j;GE[k].weight=GA[i][j];k++;}}}}/*按升序排列图的边集数组*/void SortEdge(EdgeNode GE,int m){int i,j,k;Edge temp;for(i=1;i<m;i++){k=i;for(j=i+1;j<=m;j++){if(GE[k].weight>GE[j].weight){k=j;}}if(k!=i){temp=GE[i];GE[i]=GE[k];GE[k]=temp;}}}/*利用普里姆算法从初始点v出发求邻接矩阵表示的图的最小生成树*/void Prim(adjmatrix GA,EdgeNode T){int i,j,k,min,u,m,w;Edge temp;/*给T赋初值,对应为v1依次到其余各顶点的边*/k=1;for(i=1;i<=n;i++){if(i!=1){T[k].fromvex=1;T[k].tovex=i;T[k].weight=GA[1][i];k++;}}/*进行n-1次循环,每次求出最小生成树中的第k条边*/for(k=1;k<n;k++){min=MaxNum;m=k;for(j=k;j<n;j++){if(T[j].weight<min){min=T[j].weight;m=j;}}/*把最短边对调到k-1下标位置*/temp=T[k];T[k]=T[m];T[m]=temp;/*把新加入最小生成树T中的顶点序号赋给j*/j=T[k].tovex;/*修改有关边,使T中到T外的每一个顶点保持一条到目前为止最短的边*/for(i=k+1;i<n;i++){u=T[i].tovex;w=GA[j][u];if(w<T[i].weight){T[i].weight=w;T[i].fromvex=j;}}}}/*输出边集数组的每条边*/void OutEdge(EdgeNode GE,int e){int i;printf("按照起点,终点,权值的形式输出的最小生成树为:\n");for(i=1;i<=e;i++){printf("%d,%d,%d\n",GE[i].fromvex,GE[i].tovex,GE[i].weight);}}void main(){adjmatrix GA;Edge GE[n*(n-1)/2],T[n];CreatMatrix(GA);InitEdge(GE,arcnum);GetEdgeSet(GA,GE);SortEdge(GE,arcnum);Prim(GA,T);printf("\n");OutEdge(T,n-1);}。

prim算法c语言

prim算法c语言

prim算法c语言什么是Prim算法?Prim算法,也叫普里姆算法,是一种用于求解最小生成树的贪心算法。

最小生成树是指在一个无向连通图中,连接所有节点且边权值之和最小的树。

Prim算法的基本思想是从一个起始节点开始,每次选择与当前已经构建好的部分形成的子图相连的、权值最小的边所连接的节点,并将该节点加入到已经构建好的部分中。

直到所有节点都被加入到已经构建好的部分中,此时得到了一棵最小生成树。

Prim算法步骤1. 选定一个起点作为已经构建好的部分。

2. 将与该起点相连且未被访问过的边加入到候选集合中。

3. 从候选集合中选择一条权值最小的边连接到未被访问过的节点,并将该节点加入到已经构建好的部分中。

4. 将新加入节点所连接且未被访问过的边加入到候选集合中。

5. 重复步骤3和步骤4,直至所有节点都被加入到已经构建好的部分中。

Prim算法C语言实现下面给出Prim算法C语言实现代码:```#include <stdio.h>#include <stdlib.h>#include <limits.h>#define MAX_VERTICES 100#define INF INT_MAXtypedef struct {int weight;int visited;} Vertex;typedef struct {int vertices[MAX_VERTICES][MAX_VERTICES]; int num_vertices;} Graph;void init_graph(Graph *graph, int num_vertices) {graph->num_vertices = num_vertices;for (int i = 0; i < num_vertices; i++) {for (int j = 0; j < num_vertices; j++) {graph->vertices[i][j] = INF;}}}void add_edge(Graph *graph, int u, int v, int weight) { graph->vertices[u][v] = weight;graph->vertices[v][u] = weight;}void prim(Graph *graph) {Vertex vertices[MAX_VERTICES];for (int i = 0; i < graph->num_vertices; i++) {vertices[i].weight = INF;vertices[i].visited = 0;}vertices[0].weight = 0;for (int i = 0; i < graph->num_vertices - 1; i++) {// 找到未访问过的权值最小的节点int min_vertex_index = -1;for (int j = 0; j < graph->num_vertices; j++) {if (!vertices[j].visited && (min_vertex_index == -1 || vertices[j].weight < vertices[min_vertex_index].weight)) { min_vertex_index = j;}}// 将该节点标记为已访问vertices[min_vertex_index].visited = 1;// 更新与该节点相连的未访问过的节点的权值for (int j = 0; j < graph->num_vertices; j++) {if (!vertices[j].visited && graph->vertices[min_vertex_index][j] < vertices[j].weight) {vertices[j].weight = graph->vertices[min_vertex_index][j];}}}// 输出最小生成树printf("Minimum Spanning Tree:\n");for (int i = 1; i < graph->num_vertices; i++) {printf("%d - %d (%d)\n", i, (i - 1), vertices[i].weight); }}int main() {Graph graph;init_graph(&graph, 6);add_edge(&graph, 0, 1, 6);add_edge(&graph, 0, 2, 1);add_edge(&graph, 0, 3, 5);add_edge(&graph, 1, 4, 3);add_edge(&graph, 2, 4, 5);add_edge(&graph, 2, 3, 5);add_edge(&graph, 2, 5, 4);add_edge(&graph, 3 ,5 ,2);prim(&graph);return EXIT_SUCCESS;}```代码解释- 定义了Vertex结构体,用于存储节点的权值和访问状态。

Prim算法优化策略

Prim算法优化策略

Prim算法优化策略Prim算法是一种用于求解最小生成树问题的经典算法。

它通过逐步选择与当前生成树相连的最小权值边来构造最小生成树。

在实际应用中,Prim算法的时间复杂度较高,因此需要一些优化策略来提高算法效率。

一、延迟更新策略在Prim算法中,每次选择最小权值的边添加到生成树中后,就需要更新与新增节点相邻的边的权值。

而延迟更新策略可以将这个更新过程延迟到后面再进行,避免了反复更新造成的时间浪费。

具体实现时,可以使用一个优先队列(最小堆)来存储与生成树相邻的边,每次从队列中取出权值最小的边,将其添加到生成树中,并标记其相邻节点已访问。

当队列为空时,表示所有节点都已加入生成树,算法结束。

延迟更新策略可以避免多次更新同一条边的权值,大大减少了更新操作的次数,提高了算法效率。

二、稠密图优化策略Prim算法在处理稠密图(边数接近或等于节点数的平方)时,时间复杂度较高。

为了解决这个问题,可以使用邻接矩阵来表示图,同时使用一个数组来记录每个节点到生成树的最小权值。

具体实现时,可以将邻接矩阵中的边权值初始化为一个较大的值,然后从第一个节点开始,选择与当前节点最近的未访问节点,并更新它们到生成树的最小权值。

通过这种方式,可以有效地减少对稠密图中未访问节点的搜索次数,提高算法效率。

三、堆优化策略Prim算法中,每次需要选择与当前生成树相连的最小权值边,这个过程可以通过堆来实现,以减少对边权值的搜索时间。

具体实现时,可以使用一个最小堆来存储边,堆中的每个元素都是一个包含边的两个节点和权值的数据结构。

首先将第一个节点加入生成树中,然后将其相邻边添加到堆中。

每次从堆中取出权值最小的边,将其相邻节点加入生成树,并将新的边添加到堆中。

通过使用堆结构,可以快速找到最小权值的边,提高算法的效率。

综上所述,Prim算法可以通过延迟更新策略、稠密图优化策略和堆优化策略等方法来进行优化,提高算法的效率。

在实际应用中,根据具体的问题和数据特点,选择适当的优化策略可以进一步加快算法的执行速度,提高算法的实用性和可扩展性。

求无向图的最小生成树算法——Prim与Kruskal

求无向图的最小生成树算法——Prim与Kruskal
一.Prim算法
1.算法思想
对于图G=(V,E),用Prim算法求最小生成树T=(S,TE)的流程如下
① 初始化:设S、TE为空集,任选节点K加入S。
② 选取一条权值最小的边(X,Y),其中X∈S,且not (Y∈S) 即,选取一条权值最小的、连接着S中一点与S外一点的边。
以上操作重复|V|-1次结束。由于每次加入S的点i都在当时取到了符合流程②的边min{lowcost},而lowcost[i]=w(i,closest[i]),所以此时的最小生成树的各边就是(i,closest[i]),i∈V且not (i=x)【需要注意的是出发点x的closest[x]还是x,所以应忽略,实际取到x-1条边】。把i从1取到|V|,便得到最小生成树T的每条边。
为了比较快速地选边,我们用两个数组lowcost、closest动态地维护每一个点到S的最短距离。在某一状态下,lowcost[i]表示所有与i相连且另一端点在S中的边中的权值最小值,closest[i]表示在S中且与i相连的点中与i之间距离最小的点。显然,lowcost[i]=w(i,closest[i])。需要注意的是两个数组记录的都是边而不是路径。若i没有边直接连向S,则lowcost[i]=∞。另外,若i已在S中,则lowcost[i]=0。
lowcost[j] = w[k][j];
closest[j] = k;
} //由新加入S中的k点使某些点到S的距离缩短,所以更新各点的lowcost和close= 1; i <= n; i++)
if(i != closest[i]){
设出发点为x。初始时对于任意k∈V,closest[k]=x,lowcost[k]=w(k,x)【w(i,j)表示i、j间的距离。初始化时,若两点间没有边则w(i,j)赋为一个足够大的整数(如maxint),并且所有点到自身的距离赋为0,即w(i,i)=0】

Prim算法求无向图的最小生成树

Prim算法求无向图的最小生成树
{
v1=j;
v2=k;
min=G[j][k];
}
if(!in[v2])
{
path[i][0]=v1;
path[i][1]=v2;
in[v1]=1;
in[v2]=1;
min=maxright;
}
}
if(!status)
printf("We cannot deal with it because the graph is not connected!\n");
else
{
for(i=0;i<num-1;i++)
printf("Path %d:vertex %d to vertex %d\n",i+1,path[i][0]+1,path[i][1]+1);
}
return 1;
}
本文来自:中国自学编程网(详细出处参考://7402.html
下面是源代码:
/* Prim.c
Copyright (c) 2002, 2006 by ctu_85
All Rights Reserved.
*/
/* The impact of the situation of articulation point exists can be omitted in Prim algorithm but not in Kruskal algorithm */
restart:
printf("Please enter the number of vertex(s) in the graph:\n");
scanf("%d",&num);

最小生成树算法比较Prim和Kruskal算法的优劣

最小生成树算法比较Prim和Kruskal算法的优劣

最小生成树算法比较Prim和Kruskal算法的优劣在图论中,最小生成树(Minimum Spanning Tree, MST)是指一个连通图的生成树,它的所有边的权值之和最小。

最小生成树算法是解决最小生成树问题的常用方法,而Prim算法和Kruskal算法是两种经典的最小生成树算法。

本文将比较Prim算法和Kruskal算法的优劣,为读者提供更全面的了解。

一、Prim算法Prim算法是一种贪心算法,通过逐步扩展生成树的方式来构建最小生成树。

Prim算法以一个初始节点开始,然后逐渐添加与当前生成树相连的最短边,直到生成树包含图中的所有节点为止。

以下是Prim算法的基本步骤:1. 选择任意一个节点作为初始节点,并将其加入生成树中。

2. 从生成树的节点中选择一个最短边,并将与该边相连的节点加入生成树。

3. 重复步骤2,直到生成树中包含所有节点。

相比于Kruskal算法,Prim算法在每一步只考虑一个节点,并且每次选择最短边,因此Prim算法的时间复杂度为O(V^2),其中V是图中的节点数。

二、Kruskal算法Kruskal算法也是一种贪心算法,通过按照边的权值递增的顺序来构建最小生成树。

Kruskal算法从图中所有边中选择最短边,并将其加入生成树中,同时保证生成树不形成环,直到生成树中包含所有节点为止。

以下是Kruskal算法的基本步骤:1. 对图中的所有边按照权值进行排序。

2. 依次遍历排序后的边,将权值最小的边加入生成树中,并检查是否形成环。

3. 重复步骤2,直到生成树中包含所有节点。

Kruskal算法中的关键步骤是判断是否形成环,可以使用并查集数据结构来实现。

Kruskal算法的时间复杂度为O(ElogE),其中E是图中的边数。

三、Prim算法与Kruskal算法的比较1. 时间复杂度:Prim算法的时间复杂度为O(V^2),而Kruskal算法的时间复杂度为O(ElogE)。

由于E通常小于V^2,所以Kruskal算法在大多数情况下更快。

prim算法代码

prim算法代码

prim算法代码prim算法是一种用于求解最小生成树的算法。

它通过逐步扩展生成树的边集来构建最小生成树。

具体实现过程如下:1. 输入一个无向连通图G和一个起始节点v。

2. 定义两个集合:• U集合:表示已经在生成树中的节点集合。

• V集合:表示还没有加入生成树的节点集合。

3. 从V集合中任选一个节点x,并将其加入U集合。

4. 寻找从U集合到V集合的最短横切边,并将这条边加入生成树的边集。

5. 将这条横切边的另一个顶点加入U集合,并从V集合中删除。

6. 重复执行步骤4和步骤5,直到V集合为空。

下面是该算法的Python代码实现:'''prim算法实现最小生成树输入:graph: 图的邻接矩阵,graph[i][j]表示i到j的距离,若不可达则为float('inf')返回:MST_edges: 生成树的边集合,为列表,元素是二元组(i, j),表示i和j之间连边'''def prim(graph):n = len(graph) #节点个数U = [0] * n #表示已经在生成树中的节点集合V = set(range(1, n)) #表示还没有加入生成树的节点集合 MST_edges = [] #生成树的边集合while V:min_dist, node = float('inf'), -1 #记录当前最短距离和对应的节点for u in U:for v in V:if graph[u][v] < min_dist:min_dist, node = graph[u][v], v V.remove(node)U[node] = 1for i in range(n):if i != node and i not in U:if graph[node][i] < float('inf'):MST_edges.append((node, i))return MST_edges通过调用该函数,可以得到最小生成树的边集合。

数据结构之最小生成树Prim算法

数据结构之最小生成树Prim算法

数据结构之最⼩⽣成树Prim算法普⾥姆算法介绍 普⾥姆(Prim)算法,是⽤来求加权连通图的最⼩⽣成树算法 基本思想:对于图G⽽⾔,V是所有顶点的集合;现在,设置两个新的集合U和T,其中U⽤于存放G的最⼩⽣成树中的顶点,T存放G的最⼩⽣成树中的边。

从所有uЄU,vЄ(V-U) (V-U表⽰出去U的所有顶点)的边中选取权值最⼩的边(u, v),将顶点v加⼊集合U中,将边(u, v)加⼊集合T中,如此不断重复,直到U=V为⽌,最⼩⽣成树构造完毕,这时集合T中包含了最⼩⽣成树中的所有边。

代码实现1. 思想逻辑 (1)以⽆向图的某个顶点(A)出发,计算所有点到该点的权重值,若⽆连接取最⼤权重值#define INF (~(0x1<<31)) (2)找到与该顶点最⼩权重值的顶点(B),再以B为顶点计算所有点到改点的权重值,依次更新之前的权重值,注意权重值为0或⼩于当前权重值的不更新,因为1是⼀当找到最⼩权重值的顶点时,将权重值设为了0,2是会出现⽆连接的情况。

(3)将上述过程⼀次循环,并得到最⼩⽣成树。

2. Prim算法// Prim最⼩⽣成树void Prim(int nStart){int i = 0;int nIndex=0; // prim最⼩树的索引,即prims数组的索引char cPrims[MAX]; // prim最⼩树的结果数组int weights[MAX]; // 顶点间边的权值cPrims[nIndex++] = m_mVexs[nStart].data;// 初始化"顶点的权值数组",// 将每个顶点的权值初始化为"第start个顶点"到"该顶点"的权值。

for (i = 0; i < m_nVexNum; i++){weights[i] = GetWeight(nStart, i);}for (i = 0; i < m_nVexNum; i ++){if (nStart == i){continue;}int min = INF;int nMinWeightIndex = 0;for (int k = 0; k < m_nVexNum; k ++){if (weights[k]!= 0 && weights[k] < min){min = weights[k];nMinWeightIndex = k;}}// 找到下⼀个最⼩权重值索引cPrims[nIndex++] = m_mVexs[nMinWeightIndex].data;// 以找到的顶点更新其他点到该点的权重值weights[nMinWeightIndex]=0;int nNewWeight = 0;for (int ii = 0; ii < m_nVexNum; ii++){nNewWeight = GetWeight(nMinWeightIndex, ii);// 该位置需要特别注意if (0 != weights[ii] && weights[ii] > nNewWeight){weights[ii] = nNewWeight;}}for (i = 1; i < nIndex; i ++){int min = INF;int nVexsIndex = GetVIndex(cPrims[i]);for (int kk = 0; kk < i; kk ++){int nNextVexsIndex = GetVIndex(cPrims[kk]);int nWeight = GetWeight(nVexsIndex, nNextVexsIndex);if (nWeight < min){min = nWeight;}}nSum += min;}// 打印最⼩⽣成树cout << "PRIM(" << m_mVexs[nStart].data <<")=" << nSum << ": ";for (i = 0; i < nIndex; i++)cout << cPrims[i] << "";cout << endl;}3. 全部实现#include "stdio.h"#include <iostream>using namespace std;#define MAX 100#define INF (~(0x1<<31)) // 最⼤值(即0X7FFFFFFF)class EData{public:EData(char start, char end, int weight) : nStart(start), nEnd(end), nWeight(weight){} char nStart;char nEnd;int nWeight;};// 边struct ENode{int nVindex; // 该边所指的顶点的位置int nWeight; // 边的权重ENode *pNext; // 指向下⼀个边的指针};struct VNode{char data; // 顶点信息ENode *pFirstEdge; // 指向第⼀条依附该顶点的边};// ⽆向邻接表class listUDG{public:listUDG(){};listUDG(char *vexs, int vlen, EData **pEData, int elen){m_nVexNum = vlen;m_nEdgNum = elen;// 初始化"邻接表"的顶点for (int i = 0; i < vlen; i ++){m_mVexs[i].data = vexs[i];m_mVexs[i].pFirstEdge = NULL;}char c1,c2;int p1,p2;ENode *node1, *node2;// 初始化"邻接表"的边for (int j = 0; j < elen; j ++){// 读取边的起始顶点和结束顶点p1 = GetVIndex(c1);p2 = GetVIndex(c2);node1 = new ENode();node1->nVindex = p2;node1->nWeight = pEData[j]->nWeight;if (m_mVexs[p1].pFirstEdge == NULL){m_mVexs[p1].pFirstEdge = node1;}else{LinkLast(m_mVexs[p1].pFirstEdge, node1);}node2 = new ENode();node2->nVindex = p1;node2->nWeight = pEData[j]->nWeight;if (m_mVexs[p2].pFirstEdge == NULL){m_mVexs[p2].pFirstEdge = node2;}else{LinkLast(m_mVexs[p2].pFirstEdge, node2);}}}~listUDG(){ENode *pENode = NULL;ENode *pTemp = NULL;for (int i = 0; i < m_nVexNum; i ++){pENode = m_mVexs[i].pFirstEdge;if (pENode != NULL){pTemp = pENode;pENode = pENode->pNext;delete pTemp;}delete pENode;}}void PrintUDG(){ENode *pTempNode = NULL;cout << "邻接⽆向表:" << endl;for (int i = 0; i < m_nVexNum; i ++){cout << "顶点:" << GetVIndex(m_mVexs[i].data)<< "-" << m_mVexs[i].data<< "->"; pTempNode = m_mVexs[i].pFirstEdge;while (pTempNode){cout <<pTempNode->nVindex << "->";pTempNode = pTempNode->pNext;}cout << endl;}}// Prim最⼩⽣成树void Prim(int nStart){int i = 0;int nIndex=0; // prim最⼩树的索引,即prims数组的索引char cPrims[MAX]; // prim最⼩树的结果数组int weights[MAX]; // 顶点间边的权值cPrims[nIndex++] = m_mVexs[nStart].data;// 初始化"顶点的权值数组",// 将每个顶点的权值初始化为"第start个顶点"到"该顶点"的权值。

c语言prim算法

c语言prim算法

c语言prim算法
Prim算法,又称普里姆算法,是一种用于解决最小生成树问题的经典算法。

它以图论为基础,能够在一个具有权重的连通无向图中找到一棵包含所有顶点的树,且树的权重之和最小。

算法的思路相对简单。

假设有一个无向图G,其中顶点集合为V,边集合为E。

算法从一个起始节点开始,逐步扩展生成树的规模,最终得到最小生成树。

1. 初始化:
- 创建一个空的集合T,用于存放最小生成树的边。

- 随机选择一个起始节点s,并将其加入T。

2. 重复以下步骤,直到所有顶点都加入T:
- 在图G中找到一条边e,满足e的一个端点在T中,另一个端点不在T中,并且e的权重最小。

- 将边e加入T,将其另一个端点加入T。

3. 输出最小生成树。

Prim算法的核心在于选择权重最小的边,将其加入生成树。

通过不断扩展生成树的规模,直到包含所有顶点,就能得到最小生成树。

这个算法的时间复杂度为O(V^2),其中V是顶点的数量。

在稠密图中,Prim算法的效率可能较低。

为了提高效率,可以使用最小堆等
数据结构来优化选择最小边的过程,将时间复杂度降到O(ElogV)。

Prim算法在实际应用中有着广泛的用途,如网络设计、电力传输等领域。

它能够帮助我们找到一个具有最小总成本的连接方案,从而提高资源利用效率。

Prim算法是一种解决最小生成树问题的有效算法。

通过选择权重最小的边,逐步扩展生成树的规模,最终得到一个最小总成本的树。

它的应用范围广泛,并且在实际问题中具有重要意义。

希望通过以上的介绍,能够对Prim算法有一个更深入的理解。

简述prim算法的过程

简述prim算法的过程

简述prim算法的过程Prim算法是一种用于求解加权连通图的最小生成树(MinimumSpanningTree,即最小权重生成树)算法,它的命名来源于其发明者Ralph E. Prim。

本文将介绍Prim算法的基本原理、算法的步骤、Prim算法的复杂度、Prim算法的实施及相关应用。

一、Prim算法的基本原理Prim算法是基于贪心算法原理构建的一种求解最小生成树问题算法,它的核心思想是每次添加具有最小权值的节点,以此类推迭代添加,直至所有节点都加入最小生成树。

Prim算法的一般步骤如下: 1)从图中任意选取一个节点,将其加入最小生成树;2)寻找可以添加到该最小生成树中,具有最小交叉边权值的节点,将其加入最小生成树;3)继续重复执行Step2,直至所有节点都加入最小生成树为止。

二、Prim算法的步骤Prim算法步骤包括初始化操作、选择最小权值边操作、更新最小支撑树操作三个步骤,详细流程如下:(1)初始化操作:将有向图G的每个顶点的访问标志位设置为“未访问”,同时初始化最小生成树T,将T初始化为一个空集合。

(2)选择最小权值边操作:从图G中选择一个未被访问过的顶点u,设置其访问标志位为“已访问”。

然后,选择最小权值的边<u,v>,使得v也是未被访问过的顶点。

(3)更新最小支撑树操作:将最小权值边<u,v>加入到最小支撑树T中,即把顶点u和顶点v连接起来。

重复以上步骤,直至T中的边的数量等于G中的节点的数量减一,此时T就构成了图G的最小生成树。

三、Prim算法的复杂度Prim算法的时间复杂度为O(n2),其详细计算如下:假设有n个顶点、m条边。

将n个顶点全加入最小生成树T,第一次选择最小权值边的操作需要在m条边中比较n-1次,因此,每次选择最小权值边的操作要进行O(m)次比较;由于在一次搜索算法中,每个顶点最多被访问一次,因此,共需要进行n-1次搜索;因此,Prim算法的时间复杂度是O(m*n),即O(n2)。

最小生成树(prim算法)贪心算法

最小生成树(prim算法)贪心算法

最小生成树算法Prim算法设G=(V,E)是连通带权图,V={1,2,…,n}。

构造G的最小生成树的Prim算法的基本思想是:(1)置S={1}(2)只要S是V的真子集,就作如下的贪心选择选取满足条件i ∈ S,j ∈ V-S,且c[i][j]最小的边,将顶点j添加到S中。

一直到S=V时为止。

(2)选取到的所有边恰好构成G的一棵最小生成树。

源代码://科目:算法实验4//题目:设G=(V,E)是连通带权图,V={1,2,…,n}。

构造G的最小生成树的Prim算法//作者:武叶//语言:C语言//创作时间:2012年4月14日#include"stdio.h"int point[100],key_point[100],tree[100][100]; //定义三个数组用于存放关键点和最小生成树int INT_MAX=0x7fff;void prim(int end,int V); //prim算法函数int main(){int V,E; //定义顶点数V和边数Eint i,j;int start,end,distance; //定义开始顶点start和结束顶点end,以及他们的权值distanceprintf("请输入连通带权图的边数和顶点数:");while(scanf("%d%d",&V,&E)) //开始输入你要求最小生成树的顶点数和边数{printf("\n------------------------------------");for(i=1;i<=V;i++){for(j=1;j<=V;j++)tree[i][j]=INT_MAX;}printf("\n请输入%d条边的起点和终点,以及权值。

\n",E);printf("\n----------------------------------------\n");int x=1; //用x记录输入的边数while(E--){printf("第%d条边的起点:终点:权值:",x);scanf("%d%d%d",&start,&end,&distance); //记录输入的起点、终点、权值tree[start][end]=tree[end][start]=distance;x=x+1;}prim(1,V); //调用prim计算最小生成树printf("\n");}return 0;}void prim(int end,int V){int min; //定义权值最小值minfor(int i=1;i<=V;i++){point[i]=end;key_point[i]=tree[end][i];}key_point[end]=0;for(i=2;i<=V;i++){min= INT_MAX;for(int j=1;j<=V;j++)if(key_point[j]>0 && key_point[j]<min){end=j;min=key_point[j];}printf("起点%d-->终点%d连通\n",point[end],end); //输出最小生成树的连通边key_point[end]=0;for(j=1;j<=V;j++) //继续判断条件if(tree[end][j]<key_point[j])point[j]=end,key_point[j]=tree[end][j];}}运行结果截图:答销网真情提供::文章出处::::/forum.php?mod=viewthread&tid=1533&extra=page%3D1%26filter%3Dtypeid%26typeid%3D3%26typeid%3D3。

最小生成树的方法

最小生成树的方法

最小生成树的方法最小生成树(Minimum Spanning Tree,MST)是图论中的一个重要问题。

给定一个带有权重的连通图,最小生成树指的是该连通图的一棵树,它的所有节点都被连接起来,并且树的总权重最小。

有很多不同的算法可以用来求解最小生成树问题,以下是其中的两个经典算法:Prim算法和Kruskal算法。

1. Prim算法:Prim算法是一种贪心算法,通过逐步扩展最小生成树的节点集合来构建最小生成树。

具体步骤如下:1)初始化,选择一个起始节点,并将其加入最小生成树的节点集合。

2)在待选边集合中寻找与最小生成树节点集合相连且权重最小的边,将其加入最小生成树的边集合,并将边的另一端节点加入最小生成树节点集合。

3)重复步骤2,直到最小生成树节点集合中包含了图中的所有节点。

Prim算法的时间复杂度为O(V^2),其中V是节点的个数。

优化后的Prim算法可以在O(E*log(V))的时间内完成,其中E是边的个数。

2. Kruskal算法:Kruskal算法也是一种贪心算法,通过按边的权重从小到大的顺序逐步加入到最小生成树的边集合中来构建最小生成树。

具体步骤如下:1)将图中的所有边按照权重从小到大进行排序。

2)依次从排序后的边集合中选取边,如果这条边的两个端点不在同一个连通分量中,则将这条边加入最小生成树的边集合中,并将这两个端点合并到同一个连通分量中。

3)重复步骤2,直到最小生成树的边数等于节点数减一。

Kruskal算法的时间复杂度为O(E*log(E)),其中E是边的个数。

无论是Prim算法还是Kruskal算法,它们都能够保证找到最小生成树。

在实际应用中,我们可以根据具体情况选择使用哪种算法,比如Prim算法适用于稠密图,而Kruskal算法适用于稀疏图。

最小生成树的应用十分广泛。

在通信网络设计中,最小生成树可以用来找到连接所有节点的最短路径,降低网络的通信成本。

在电力系统设计中,最小生成树可以用来确定最优的输电线路,提高电力系统的稳定性。

普里姆算法和克鲁斯卡尔算法

普里姆算法和克鲁斯卡尔算法

普里姆算法和克鲁斯卡尔算法普里姆算法和克鲁斯卡尔算法介绍在图论中,最小生成树是一种重要的概念。

最小生成树是指在一个加权连通图中,找到一棵生成树,使得所有边的权值之和最小。

其中,Prim算法和Kruskal算法是两种常用的求解最小生成树的方法。

Prim算法Prim算法是一种贪心算法,它从一个顶点开始构建最小生成树。

具体实现过程如下:1. 选取任意一个顶点作为起始点,并将其标记为已访问。

2. 遍历与该顶点相邻的所有边,并将这些边加入一个优先队列中。

3. 从优先队列中选取一条权值最小的边,并将与之相邻的未被访问过的顶点标记为已访问。

4. 重复步骤2和3,直到所有顶点都被访问过。

代码实现```def prim(graph, start):visited = set()visited.add(start)edges = []min_span_tree_cost = 0while len(visited) != len(graph.keys()):candidate_edges = []for node in visited:for edge in graph[node]:if edge[1] not in visited:candidate_edges.append(edge)sorted_edges = sorted(candidate_edges, key=lambda x:x[2]) min_edge = sorted_edges[0]edges.append(min_edge)min_span_tree_cost += min_edge[2]visited.add(min_edge[1])return edges, min_span_tree_cost```时间复杂度Prim算法的时间复杂度为O(ElogV),其中E为边数,V为顶点数。

因为每次需要从优先队列中选取一条权值最小的边,所以需要使用堆来实现优先队列。

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

1.如下图,根结点为a,给出Prim算法求解最小生成树的伪代码,在下图中标出最小生成树,给出用binary min-heap来表示min-priority queue时,Prim算法的时间复杂度。

解:伪代码如下:
MST_PRIM(G, w, r) //r=a
for each u∈V[G]
do key[u]←∞
π[u]←NIL
key[r]←Q
Q←V[G]
while Q≠φ
do u←EXTRACT-MIN(Q)
for each v∈Adj[u]
do if v∈Q and w(u,v)<key[v]
thenπ[v]←u
key[v]←w(u,v)
时间复杂度分析:
如果用二叉最小堆来实现最小优先队列Q,则可以用过程BUILD-MIN-HEAP来实现程序初始化部分,其运行时间为O(V)。

while循环的循环体需执行|V|次,且由于每次EXTRACT-MIN操作需要O(lgV),所以对EXTRACT-MIN的全部调用所占用的时间为O(VlgV)。

while循环体中的for循环总共要执行O(E)次,for循环内部最后一行的赋值语句隐含了一个对最小堆进行的DECREASE-KEY操作,该操作在二叉最小堆上可以用O(lgV)时间完成。

因此,Prim算法的整个运行时间为O(VlgV+ElgV)=O(ElgV)。

相关文档
最新文档