HDU 3078 Network LCA

题意:n个点 m个询问,下面一行是n 个点的权值 再下面n-1行是双向的边

然后m个询问:k u v 若k==0,则把u点的权值改为v,否则回答u->v之间最短路经过点的权值中  第k大的值是多少

木有AC。。勿扔OJ

可以拿来学习RMQ

思路:跑个RMQ  求出LCA(u,v) 然后只要登山坡一遍就得到u->v的点,记下这些点的权值,再排个序就有第k大的数了

下面附几个测试案例和答案

#include <stdio.h>
#include <string.h>
#include <algorithm>
using namespace std;
#define N 80010 //最多有N个点
struct node{
	int from,to,nex;
}edge[N];
int head[N],edgenum;
int a[N],pre[N],temp[N];//a[]是点的权值,temp是用来记录登山坡经过的点的权值
//pre记录点的前驱,用来登山坡
int E[N*2],R[N],D[N*2],en;//R[i]代表i点第一次搜到的位置 E[i]代表搜的第i次点的编号 D[i]代表搜的第i次点的深度

void add(int u,int v){//邻接表
	edge[edgenum].from=u;	edge[edgenum].to=v;
	edge[edgenum].nex=head[u]; head[u]=edgenum++;
}
void DFS(int x,int deep){
	E[en]=x;D[en]=deep;R[x]=en++;
	for(int i=head[x];i!=-1;i=edge[i].nex)
	{
		int v=edge[i].to;
		if(R[v]==-1)
		{
			pre[v]=x;
			DFS(v,deep+1);
			E[en]=x; D[en++]=deep;
		}
	}
}
int LCA(int u,int v){
	int deep=N,t=u;
	if(R[u]>R[v]){t=u;u=v;v=t;}
	for(int i=R[u];i<=R[v];i++)
		if(deep>D[i])
			deep=D[i],t=i;
	return E[t];
}

bool cmp(int a,int b){return a>b;}
void Findans(int u,int v,int k){
	int lca=LCA(u,v),top=0;

	temp[top++]=a[lca];

	for(;u!=lca;u=pre[u])temp[top++]=a[u];//登山坡到LCA处
	for(;v!=lca;v=pre[v])temp[top++]=a[v];

	if(top<k)printf("invalid request!\n");
	else 
	{
		sort(temp,temp+top,cmp);
		printf("%d\n",temp[k-1]);
	}
}
void Init(){
	memset(head,-1,sizeof(head));
	edgenum=0;	
	memset(R,-1,sizeof(R)); en=0;  
	memset(pre,-1,sizeof(pre));
	//pre[1]=-1;	
}

int main(){
	int u,v,n,m;//n个点,从1-n
	scanf("%d%d",&n,&m);
	for(int i=1;i<=n;i++)scanf("%d",&a[i]);
	Init();
	while(--n)
	{
		scanf("%d %d",&u,&v);
		add(u,v);	add(v,u);//双向边
	}
	DFS(1,0);//建以1为根的树 此时深度为0
	while(m--)//询问
	{
		int k;	scanf("%d%d%d",&k,&u,&v);
		if(k==0)a[u]=v;
		else Findans(u,v,k);
	}
return 0;
}
/*
8 99
5 2 3 4 5 6 7 8
1 3
1 2
3 4 
3 5 
5 6
5 7
7 8
1 2 8
2 2 8
3 2 8
4 2 8
5 2 8
6 2 8
7 2 8
1 4 8
4 2 3
5 3 8
2 3 8
0 3 8
2 3 8
3 3 8
2 4 5
1 4 1
2 4 1
3 4 1

ans:
8
7
5
5
3
2
in...
8
in...
in...
7
8
7
5
8
5
4

////
1 1
5
1 1 1

ans:
5
////
2 2
2 1
1 2
2 2 1
1 1 2

ans:
1
2

11 3
1 2 3 4 5 6 7 8 9 10 11
1 4 
1 3
2 1
4 5
6 4
3 7
2 9
10 2
10 11
7 8
3 11 8
3 11 9
5 10 5

ans:
8
9
1
////
4 99
1 2 3 4
2 1
3 1
4 1
1 2 4
2 2 4
3 2 4
4 2 4
0 1 100
1 2 4
2 2 4
3 2 4
4 2 4
1 3 3
1 3 1
2 3 1


ans:
4
2
1
in...
100
4
2
in...
3
100
3


//RMQ
//http://blog.csdn.net/liang5630/article/details/7917702
*/


你可能感兴趣的:(网络编程,LCA)