zoukankan      html  css  js  c++  java
  • BZOJ 1787 裸LCA

    一个求树上LCA的裸题。。WC(伪。。)里想出来的。。

    题目大意:给出树上的三个点,要求确定一个集合点,使得这三个点到集合点的路径权值和最小。所有边权均为1。

    先考虑两个点A、B的情形。。显然这两个点间路径上的任何一点都可以作为集合点。。

    然后再加入第三个点C。。画个图不难证明此时最优集合点应是LCA(A, B)。

    但是A、B、C分别是哪个点呢?。。

    枚举取最优解。。

    但看了黄学长博客,给出了两种解法。。一种就是枚举,另一种则给出了一句结论,私认为很有道理。。

    求出两两lca,其中有两个相同,答案则为另一个,画画图就可以理解”

    这样的话,就可以直接确定是那一个LCA。。不用判断了。。

    所以答案就很好搞了。。求求LCA加加就行了。。

    用的是黄学长的模板。。

    // BZOJ 1787
    
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    
     typedef long long LL;
     typedef unsigned long long uLL;
     const int N=500000+5, INF=0x3f3f3f3f;
    
     #define rep(i,a,b) for (int i=a; i<=b; i++)
     #define dep(i,a,b) for (int i=a; i>=b; i--)
     #define read(x) scanf("%d", &x)
     #define fill(a,x) memset(a, x, sizeof(a))
    
     int n, q, dep[N], last[N], fa[N][20];
    
     struct Edge {
     	int to, pre;
     } e[N*2];  // 注意无向边数要乘2!
    
     int k=0;
     void ine(int x, int y) {
     	k++;
     	e[k].to=y; e[k].pre=last[x]; last[x]=k;
     	k++;
     	e[k].to=x; e[k].pre=last[y]; last[y]=k;
     }
     #define reg(i,a) for (int i=last[a]; i; i=e[i].pre)
    
     void change(int x, int dad) {  // 无根树转有根树的同时递推每个点的2^i祖先
     	rep(i,1,18) {
     		if (dep[x]<(1<<i)) break;
     		fa[x][i]=fa[fa[x][i-1]][i-1];
     	}
     	reg(i,x) {
     		int y=e[i].to;
     		if (y==dad) continue;
     		dep[y]=dep[x]+1;
     		fa[y][0]=x;
     		change(e[i].to, x);
     	}
     }
    
     int LCA(int x, int y) {
     	if (dep[x]<dep[y]) swap(x, y);
     	int d=dep[x]-dep[y];
     	rep(i,0,18) 
     		if ((1<<i)&d) x=fa[x][i];
     	dep(i,18,0) 
     		if (fa[x][i]!=fa[y][i]) { x=fa[x][i]; y=fa[y][i]; }
     	if (x==y) return x; else return fa[x][0]; 
     }
    
     int dis(int x, int y) { int t=LCA(x,y); return dep[x]+dep[y]-2*dep[t]; }
    
     void solve(int x, int y, int z) {
     	int p1=LCA(x, y), p2=LCA(x, z), p3=LCA(y, z), p, ans;
     	if (p1==p2) p=p3; else if (p2==p3) p=p1; else p=p2;
     	ans=dis(x, p)+dis(y, p)+dis(z, p);
     	printf("%d %d
    ", p, ans);
     }
    
    int main()
    {
    	read(n); read(q);
    	rep(i,1,n-1) {
    		int u, v;
    		read(u); read(v);
    		ine(u, v);
    	}
    	change(1, 0);
    	rep(i,1,q) {
    		int x, y, z;
    		read(x); read(y); read(z);
    		solve(x, y, z);
    	}
    	
    	return 0;
    }
    



  • 相关阅读:
    i春秋——春秋争霸write up
    2017-2018-2 《网络对抗技术》20155322 Exp6 信息搜集与漏洞扫描
    2017-2018-2 《网络对抗技术》 20155322 Exp 5 MSF基础应用
    OpenCV学习系列(一) Mac下OpenCV + xcode人脸检测实现
    2017-2018-2 《网络对抗技术》 20155322 Exp4 恶意代码分析
    MacOS下netstat和lsof使用的若干问题
    OpenCV学习系列(零) Mac下OpenCV + xcode环境搭建
    2017-2018-2 《网络对抗技术》 20155322 Exp3 免杀原理与实践
    2017-2018-2 《网络对抗技术》 20155322 第五周 Exp2 后门原理与实践
    信息安全铁人三项赛--资质赛writeup
  • 原文地址:https://www.cnblogs.com/yearwhk/p/5119872.html
Copyright © 2011-2022 走看看