zoukankan      html  css  js  c++  java
  • 【POJ 3764】The xorlongest Path【Trie】

    题目大意:

    题目链接:http://poj.org/problem?id=3764
    在一棵树中选择任意两个结点,使得他们之间路径之和最大。


    思路:

    首先我们设点1为根节点,然后求每个节点与根节点(点1)的路径的异或值。那么设d[i]d[i]为点1到点ii之间路径异或值,那么必然有d[i]=d[father]  xor  dis[i][father]d[i]=d[father]\ \ xor \ \ dis[i][father]
    那么在根据a  xor  a=0a\ \ xor \ \ a=0,那么很容易得到点ii和点jj的异或值为d[i]  xor  d[j]d[i]\ \ xor \ \ d[j],因为如果它们在根结点的两侧,那么上述公式十分显然。如果他们在同一侧,那么若有一条重复路径ii,那么就绝对会有d[i]  xor  d[i]d[i]\ \ xor \ \ d[i],那么就为0,相当于低抵消了。
    那么题目就变成了求一个数列中的任意两个数字亦或的最大值,就根这道题完全一样了。


    代码:

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #define N 200005
    #define M 3000005
    #define up 30
    using namespace std;
    
    int n,x,y,k,tot=1,head[N],trie[M][2];
    long long d[N],z,ans,sum;
    
    struct edge  //邻接表
    {
    	int to,next;
    	long long dis;
    }e[N];
    
    void add(int from,int to,long long dis)  //建图(树)
    {
    	k++;
    	e[k].to=to;
    	e[k].dis=dis;
    	e[k].next=head[from];
    	head[from]=k;
    }
    
    void dfs(int x,int fa)  //深搜求点1和点i的异或距离
    {
    	int v;
    	for (int i=head[x];i;i=e[i].next)
    	{
    		v=e[i].to;
    		if (v==fa) continue;  //不能到父节点
    		d[v]=d[x]^e[i].dis;  //求距离
    		dfs(v,x);
    	} 
    }
    
    void insert(long long x)  //插入
    {
    	int p=1;
    	for (int i=up;i>=0;i--)
    	{
    		int id=(x>>i)&1;
    		if (!trie[p][id]) trie[p][id]=++tot;
    		p=trie[p][id];
    	}
    }
    
    void find(long long x)  //查找
    {
    	int p=1;
    	for (int i=up;i>=0;i--)
    	{
    		int id=(x>>i)&1;
    		if (trie[p][id^1])
    		{
    			sum=sum*2+1;
    			p=trie[p][id^1];
    		}
    		else
    		 if (trie[p][id])
    		 {
    		 	sum=sum*2;
    		 	p=trie[p][id];
    		 }
    		//else return;
    	}
    }
    
    int main()
    {
    	while (scanf("%d",&n)==1)
    	{
    		memset(head,0,sizeof(head));
    		memset(d,0,sizeof(d));
    		memset(trie,0,sizeof(trie));
    		tot=1;
    		k=0;
    		ans=0;  //初始化
    		for (int i=1;i<n;i++)
    		{
    			scanf("%d%d%lld",&x,&y,&z);
    			x++;
    			y++;
    			add(x,y,z);
    			add(y,x,z);
    		}
    		dfs(1,-1);
    		for (int i=1;i<=n;i++) insert(d[i]);
    		for (int i=1;i<=n;i++)
    		{
    			sum=0;
    			find(d[i]);
    			ans=max(ans,sum);
    		}
    		printf("%lld\n",ans);
    	}
    	return 0;
    }
    
  • 相关阅读:
    根据不同运行环境配置和组织node.js应用
    python下pip使用bug汇总
    python的虚拟环境
    nginx使用手册--nginx的命令行参数
    nginx使用手册--nginx.conf文件配置详解
    使用async读取异步数据
    [Bootstrap]组件(三)
    [Bootstrap]组件(二)
    [Bootstrap]组件(一)
    [javascript|基本概念|一元操作符]学习笔记
  • 原文地址:https://www.cnblogs.com/hello-tomorrow/p/11998700.html
Copyright © 2011-2022 走看看