NewTrain1 T9:[WC2011]最大XOR和路径

题目分析

我们把选出的路径看做一条1到n的简单路径+一些环

简单路径可以任取一条,就算我们选出的这条不是最优解的路径,我们也可以认为,我们走这条路径到了n,又走最优解的路径回到1,然后再走这条路径到n,这样其实就是这条路径+一个环,异或一下就抵消了这条路径。

那么对于一个不直接与这条路径联通的环,我们也可以认为这个环可以异或到答案里面。因为从这条路一条分岔出去到这个环,然后再原路返回,那从这条路到环的那段分岔异或一下就抵消掉了,所以可以直接异或这个环。

那么解题思路就是,预处理出所有的环,将其加入线性基中。然后再计算出1到n的任意一条简单路径,长度记为d,我们在线性基中询问与d异或的最大值即可。

预处理环与计算简单路径可以一并在dfs中完成。

 1 #include<bits/stdc++.h>
 2 #define INTMAX 2147483647LL
 3 #define PII pair<int,int>
 4 #define MK make_pair
 5 #define re register
 6 using namespace std;
 7 typedef long long ll;
 8 const double Pi=acos(-1.0);
 9 const int Inf=0x3f3f3f3f;
10 const int MAXN=2e5+10;
11
12 inline int read(){
13     re int x=0,f=1,ch=getchar();
14     while(!isdigit(ch))f=ch==‘-‘?-1:1,ch=getchar();
15     while(isdigit(ch))x=x*10+ch-48,ch=getchar();
16     return x*f;
17 }
18 inline ll readll(){
19     re ll x=0,f=1,ch=getchar();
20     while(!isdigit(ch))f=ch==‘-‘?-1:1,ch=getchar();
21     while(isdigit(ch))x=x*10+ch-48,ch=getchar();
22     return x*f;
23 }
24
25 struct Edge{
26     int to,nxt;
27     ll d;
28 }e[MAXN<<1];
29 int cnt,head[MAXN];
30 int n,m;
31 bool vis[MAXN];
32 ll dis[MAXN],bse[70];
33 inline void add_edge(int u,int v,ll d){
34     e[++cnt].to=v;e[cnt].d=d;e[cnt].nxt=head[u];head[u]=cnt;
35 }
36
37 inline void Insert(ll x){
38     for(int i=60;i>=0;--i){
39         if(x&(1ll<<i)){
40             if(bse[i]) x^=bse[i];
41             else{
42                 bse[i]=x;
43                 break;
44             }
45         }
46     }
47 }
48 inline ll Query(ll x){
49     for(int i=60;i>=0;--i){
50         if((x^bse[i])>x) x^=bse[i];
51     }
52     return x;
53 }
54 inline void dfs(int x){
55     vis[x]=true;
56     for(int i=head[x],y;i;i=e[i].nxt){
57         y=e[i].to;
58         if(vis[y]) Insert(dis[x]^dis[y]^e[i].d);
59         else{
60             dis[y]=dis[x]^e[i].d;
61             dfs(y);
62         }
63     }
64 }
65 int main(){
66     n=read();m=read();
67     for(int i=1;i<=m;++i){
68         int u=read(),v=read();
69         ll d=readll();
70         add_edge(u,v,d);
71         add_edge(v,u,d);
72     }
73     dfs(1);
74     printf("%lld\n",Query(dis[n]));
75     return 0;
76 }

原文地址:https://www.cnblogs.com/LI-dox/p/11267300.html

时间: 2024-10-31 11:58:57

NewTrain1 T9:[WC2011]最大XOR和路径的相关文章

[WC2011]最大XOR和路径 线性基

题面 题面 题解 其实是一个很重要的套路啦. 首先我们从s到t的一个基础路径肯定是一条链,在此基础上,我们唯一可以带来一些增益的走法就是在走这条链的基础上走一些环,因为xor的特点,来回走的路都相当于没走,而只有环可以做到不往回走却能回到原点. 因此只有走环才会给原来的路线带来改变,否则走了都等于没走. 因此我们将图上所有简单环异或后的01串加入线性基. 那么对于一条指定的链,所以环可以带给它的最大增益可以用类似求最大异或和的方式来求. 所以我们还需要枚举每一条链? 其实不用. 因为所有链的起点

[luogu4151 WC2011] 最大XOR和路径 (线性基)

传送门 输入输出样例 输入样例#1: 5 7 1 2 2 1 3 2 2 4 1 2 5 1 4 5 3 5 3 4 4 3 2 输出样例#1: 6 说明 [样例说明] 根据异或的性质,将一个数异或两次便会消除影响 那么预处理所有环插入线性基中,之后随便(因为能够消除影响)找一条简单路径查询最大值即可 code: //By Menteur_Hxy #include<cstdio> #include<iostream> #include<cstring> #include

p4151 [WC2011]最大XOR和路径

分析 我们发现任取条路径 对于路径外的环一定可以将它完整的取到 而对于和路径有交的环相当于用一段新路径代替原来的一段路径 所以我们只需求出任意一个1到n的路径和图上所有环的值 然后借助线性基求出异或最大值即可 代码 #include<iostream> #include<cstdio> #include<cstring> #include<string> #include<algorithm> #include<cctype> #in

【概率DP/高斯消元】BZOJ 2337:[HNOI2011]XOR和路径

2337: [HNOI2011]XOR和路径 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 682  Solved: 384[Submit][Status][Discuss] Description 几乎是一路看题解过来了.. 拖了一个星期的题目- - 已然不会概率DP(说得好像什么时候会过一样),高斯消元(打一次copy一遍). 发现异或题目的新解决方法:按位处理.. 发现DP新方法:高斯消元. f[k][i]代表第k位权值起点为i到终点时答案

【BZOJ 2337】 2337: [HNOI2011]XOR和路径(概率DP、高斯消元)

2337: [HNOI2011]XOR和路径 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1170  Solved: 683 Description Input Output Sample Input Sample Output HINT Source Day2 [分析] 这题终于自己打出来了高斯消元.没有对比代码了... 很心酸啊..调试的时候是完全没有方向的,高斯消元还要自己一步步列式子然后消元解..[为什么错都不知道有时候 这题显然是不能

【BZOJ2337】[HNOI2011]XOR和路径 期望DP+高斯消元

[BZOJ2337][HNOI2011]XOR和路径 Description 题解:异或的期望不好搞?我们考虑按位拆分一下. 我们设f[i]表示到达i后,还要走过的路径在当前位上的异或值得期望是多少(妈呀好啰嗦),设d[i]表示i的度数.然后对于某条边(a,b),如果它的权值是1,那么f[b]+=(1-f[a])/d[a]:如果它的权值是0,那么f[b]+=f[a]/d[a],然后我们移个项,就变成了一堆方程组求解,直接高斯消元. 别忘了f[n]=0! #include <cstdio> #i

【BZOJ2337】Xor和路径(高斯消元)

[BZOJ2337]Xor和路径(高斯消元) 题面 BZOJ 题解 我应该多学点套路: 对于xor之类的位运算,要想到每一位拆开算贡献 所以,对于每一位拆开来看 好了,既然是按位来算 我们就只需要计算从\(1\)号点开始 到\(n\)的路径中,路径的异或和为\(1\)的概率 显然没法算 还是一样的 考虑高斯消元 对于每一个节点\(i\) \[f[i]=\sum_{w(u,i)=1}\frac{1-f[u]}{op[u]}+\sum_{w(u,i)=1}\frac{f[u]}{op[u]}\] 其

【BZOJ】2337: [HNOI2011]XOR和路径

[算法]期望+高斯消元 [题解]因为异或不能和期望同时运算,所以必须转为加乘 考虑拆位,那么对于边权为1取反,边权为0不变. E(x)表示从x出发到n的路径xor期望. 对于点x,有E(x)=Σ(1-E(y))(边权1)||E(y)(边权0)/t[x]  t[x]为x的度. 那么有n个方程,整体乘上t[x]确保精度,右项E(x)移到左边--方程可以各种变形. 每次计算完后*(1<<k)就是贡献. 逆推的原因在于n不能重复经过,而1能重复经过,所以如果计算"来源"不能计算n,

xor和路径(codevs 2412)

题目描述 Description 给定一个无向连通图,其节点编号为1到N,其边的权值为非负整数.试求出一条从1号节点到 N 号节点的路径,使得该路径上经过的边的权值的"XOR 和"最大.该路径可以重复经过某些节点或边,当一条边在路径中出现多次时,其权值在计算"XOR 和"时也要被重复计算相应多的次数. 直接求解上述问题比较困难,于是你决定使用非完美算法.具体来说,从1号节点开始,以相等的概率,随机选择与当前节点相关联的某条边,并沿这条边走到下一个节点,重复这个过程,