[bzoj4568] [loj#2013] [Scoi2016] 幸运数字

Description

\(A\) 国共有 \(n\) 座城市,这些城市由 \(n-1\) 条道路相连,使得任意两座城市可以互达,且路径唯一。每座城市都有一个幸运数字,以纪念碑的形式矗立在这座城市的正中心,作为城市的象征。一些旅行者希望游览 \(A\) 国。旅行者计划乘飞机降落在 \(x\) 号城市,沿着 \(x\) 号城市到 \(y\) 号城市之间那条唯一的路径游览,最终从 \(y\) 城市起飞离开 \(A\) 国。在经过每一座城市时,游览者就会有机会与这座城市的幸运数字拍照,从而将这份幸运保存到自己身上。然而,幸运是不能简单叠加的,这一点游览者也十分清楚。他们迷信着幸运数字是以异或的方式保留在自己身上的。例如,游览者拍了 3 张照片,幸运值分别是 5,7,11,那么最终保留在自己身上的幸运值就是 9(5 \(xor\) 7 \(xor\) 11)。有些聪明的游览者发现,只要选择性地进行拍照,便能获得更大的幸运值。例如在上述三个幸运值中,只选择 5 和 11 ,可以保留的幸运值为 14 。现在,一些游览者找到了聪明的你,希望你帮他们计算出在他们的行程安排中可以保留的最大幸运值是多少。

Input

第一行包含 \(2\) 个正整数 \(n\)\(q\),分别表示城市的数量和旅行者数量。第二行包含 \(n\) 个非负整数,其中第 \(i\) 个整数 \(Gi\) 表示 \(i\) 号城市的幸运值。随后 \(n-1\) 行,每行包含两个正整数 \(x\)\(y\),表示 \(x\) 号城市和 \(y\) 号城市之间有一条道路相连。随后 \(q\) 行,每行包含两个正整数 \(x\)\(y\),表示这名旅行者的旅行计划是从 \(x\) 号城市到 \(y\) 号城市。
\(N \leq20000,Q \leq 200000,Gi \leq 2^60\)

Output

输出需要包含 \(q\) 行,每行包含 \(1\) 个非负整数,表示这名旅行者可以保留的最大幸运值。

Sample Input

4 2

11 5 7 9

1 2

1 3

1 4

2 3

1 4

Sample Output

14

11


想法

要求一堆数中的最大异或和要用线性基。
那么这道题我们就要把一条路径上的所有店的值扔到线性基里去求最大。

怎么扔进去呢?倍增 ,维护每个点网上跳 \(2^i\) 个点这条链上的线性基
查询的时候 \(lca\) 的时候合并一下线性基就可以了。

没有卡常,极其感动。


代码

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;

const int N = 20005;
typedef long long ll;

ll lread(){
    ll x=0;
    char ch=getchar();
    while(!isdigit(ch)) ch=getchar();
    while(isdigit(ch)) x=x*10+ch-'0',ch=getchar();
    return x;
}
int read(){
    int x=0;
    char ch=getchar();
    while(!isdigit(ch)) ch=getchar();
    while(isdigit(ch)) x=x*10+ch-'0',ch=getchar();
    return x;
}

int n;
ll G[N];

struct node{
    int v;
    node *nxt;
}pool[N*2],*h[N];
int cnt;
void addedge(int u,int v){
    node *p=&pool[++cnt],*q=&pool[++cnt];
    p->v=v;p->nxt=h[u];h[u]=p;
    q->v=u;q->nxt=h[v];h[v]=q;
}

int f[N][16],dep[N];
ll g[N][16][65];

void ins(ll *A,ll x) {
    for(int i=60;i>=0;i--)
        if(x&(1ll<<i)){
            if(!A[i]) { A[i]=x; return; }
            x^=A[i];
        }
}
void merge(ll *A,ll *B) { for(int i=60;i>=0;i--) if(B[i]) ins(A,B[i]); }
ll MAX(ll *A) {
    ll ret=0;
    for(int i=60;i>=0;i--) ret=max(ret,ret^A[i]);
    return ret;
}

void dfs(int u){
    int v;
    for(node *p=h[u];p;p=p->nxt)
        if(!dep[v=p->v]){
            dep[v]=dep[u]+1;
            f[v][0]=u; ins(g[v][0],G[u]);
            for(int j=1;j<16;j++){
                f[v][j]=f[f[v][j-1]][j-1];
                memcpy(g[v][j],g[v][j-1],sizeof(g[v][j]));
                merge(g[v][j],g[f[v][j-1]][j-1]);
            }
            dfs(v);
        }
}
ll lca(int x,int y){
    ll c[65]; memset(c,0,sizeof(c));
    ins(c,G[x]); ins(c,G[y]);
    if(dep[x]<dep[y]) swap(x,y);
    for(int i=15;i>=0;i--)
        if(dep[f[x][i]]>=dep[y]) merge(c,g[x][i]),x=f[x][i];
    if(x==y) return MAX(c);
    for(int i=15;i>=0;i--)
        if(f[x][i]!=f[y][i]){
            merge(c,g[x][i]); x=f[x][i];
            merge(c,g[y][i]); y=f[y][i];
        }
    merge(c,g[x][0]);
    return MAX(c);
}

int main()
{
    int Q;
    n=read(); Q=read();
    for(int i=1;i<=n;i++) G[i]=lread();
    for(int i=1;i<n;i++) addedge(read(),read());
    
    dep[1]=1; dfs(1);
    
    while(Q--)
        printf("%lld\n",lca(read(),read()));
    
    return 0;
}

转载于:https://www.cnblogs.com/lindalee/p/11431943.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
4S店客户管理小程序-毕业设计,基于微信小程序+SSM+MySql开发,源码+数据库+论文答辩+毕业论文+视频演示 社会的发展和科学技术的进步,互联网技术越来越受欢迎。手机也逐渐受到广大人民群众的喜爱,也逐渐进入了每个用户的使用。手机具有便利性,速度快,效率高,成本低等优点。 因此,构建符合自己要求的操作系统是非常有意义的。 本文从管理员、用户的功能要求出发,4S店客户管理系统中的功能模块主要是实现管理员服务端;首页、个人中心、用户管理、门店管理、车展管理、汽车品牌管理、新闻头管理、预约试驾管理、我的收藏管理、系统管理,用户客户端:首页、车展、新闻头、我的。门店客户端:首页、车展、新闻头、我的经过认真细致的研究,精心准备和规划,最后测试成功,系统可以正常使用。分析功能调整与4S店客户管理系统实现的实际需求相结合,讨论了微信开发者技术与后台结合java语言和MySQL数据库开发4S店客户管理系统的使用。 关键字:4S店客户管理系统小程序 微信开发者 Java技术 MySQL数据库 软件的功能: 1、开发实现4S店客户管理系统的整个系统程序; 2、管理员服务端;首页、个人中心、用户管理、门店管理、车展管理、汽车品牌管理、新闻头管理、预约试驾管理、我的收藏管理、系统管理等。 3、用户客户端:首页、车展、新闻头、我的 4、门店客户端:首页、车展、新闻头、我的等相应操作; 5、基础数据管理:实现系统基本信息的添加、修改及删除等操作,并且根据需求进行交流信息的查看及回复相应操作。
现代经济快节奏发展以及不断完善升级的信息化技术,让传统数据信息的管理升级为软件存储,归纳,集中处理数据信息的管理方式。本微信小程序医院挂号预约系统就是在这样的大环境下诞生,其可以帮助管理者在短时间内处理完毕庞大的数据信息,使用这种软件工具可以帮助管理人员提高事务处理效率,达到事半功倍的效果。此微信小程序医院挂号预约系统利用当下成熟完善的SSM框架,使用跨平台的可开发大型商业网站的Java语言,以及最受欢迎的RDBMS应用软件之一的MySQL数据库进行程序开发。微信小程序医院挂号预约系统有管理员,用户两个角色。管理员功能有个人中心,用户管理,医生信息管理,医院信息管理,科室信息管理,预约信息管理,预约取消管理,留言板,系统管理。微信小程序用户可以注册登录,查看医院信息,查看医生信息,查看公告资讯,在科室信息里面进行预约,也可以取消预约。微信小程序医院挂号预约系统的开发根据操作人员需要设计的界面简洁美观,在功能模块布局上跟同类型网站保持一致,程序在实现基本要求功能时,也为数据信息面临的安全问题提供了一些实用的解决方案。可以说该程序在帮助管理者高效率地处理工作事务的同时,也实现了数据信息的整体化,规范化与自动化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值