线性基【CF845G】Shortest Path Problem?
Description
给定一张 \(n\) 个点 \(m\) 条边的无向图,一开始你在点 \(1\),且价值为 \(0\)
每次你可以选择一个相邻的点,然后走过去,并将价值异或上该边权
如果在点 \(n\),你可以选择结束游戏
求一种方案,使得结束游戏后价值最小
\(n,m \le 10^5\)
Input
第一行为两个整数\(n,m\)代表有\(n\)个点\(m\)条边。
接下来\(m\)行,描述一条从\(x\)到\(y\)长度为\(z\)的无向边。
Output
一个整数,代表最小价值。
首先,很明确的一点,题目要求我们求出很多条边的最小异或和。
由此,我们可以想到线性基
由于我们可以重复经过一些边,那么根据异或性质,当这条边被重复走过两次,那它对答案的贡献就是\(0\)。但是即使这样,它还可能连向其他的点。
虽然我们没办法枚举边,但是可以考虑将这些边所在分为两种。
- 环上的边
- 链上的边
但是我们通向一个环的时候会经过一条连向这个环的边两次。(一进一出)。
因此,我们考虑维护每个环的异或和,塞入线性基中。
再找一条链,去和其异或起来最小。即可。
这条链可以随便选择。
简单证明一下;
假设存在两条链\(A,B\),我们现在选择了不优的\(A\)链,但是在求解答案的时候(利用线性基)
我们会异或到一个环(\(A,B\)链围成的环),这时,就好比我们原路返回,又选择了较优的\(B\)链。
因此,这个题就解决了.
代码
#include<cstdio>
#include<algorithm>
#include<iostream>
#define R register
#define lo long long
using namespace std;
const int gz=1e5+8;
inline void in(R int &x)
{
int f=1;x=0;char s=getchar();
while(!isdigit(s)){if(s=='-')f=-1;s=getchar();}
while(isdigit(s)){x=x*10+s-'0';s=getchar();}
x*=f;
}
int head[gz<<1],tot;
struct cod{int u,v;lo w;}edge[gz<<2];
lo p[64],dis[gz];
inline void add(R int x,R int y,R lo z)
{
edge[++tot].u=head[x];
edge[tot].v=y;
edge[tot].w=z;
head[x]=tot;
}
int n,m;
bool vis[gz];
inline void ins(R lo x)
{
for(R int i=63;i>=0;i--)
{
if((x>>i)&1LL)
{
if(p[i])
x^=p[i];
else
{
p[i]=x;
break;
}
}
}
}
inline lo query(R lo o)
{
R lo res=o;
for(R int i=63;i>=0;i--)
if((res^p[i])<res)res^=p[i];
return res;
}
void dfs(R int u,R lo now)
{
vis[u]=true;dis[u]=now;
for(R int i=head[u];i;i=edge[i].u)
{
if(!vis[edge[i].v])
dfs(edge[i].v,now^edge[i].w);
else ins(now^edge[i].w^dis[edge[i].v]);
}
}
int main()
{
in(n);in(m);
for(R int i=1,x,y;i<=m;i++)
{
R lo z;
in(x),in(y);
scanf("%lld",&z);
add(x,y,z),add(y,x,z);
}
dfs(1,0);
printf("%lld\n",query(dis[n]));
}
线性基【CF845G】Shortest Path Problem?的更多相关文章
- [CF845G]Shortest Path Problem?
题目大意:同这道题,只是把最大值变成了最小值 题解:略 卡点:无 C++ Code: #include <cstdio> #define maxn 100010 #define maxm ...
- Codefroces Educational Round 27 845G Shortest Path Problem?
Shortest Path Problem? You are given an undirected graph with weighted edges. The length of some pat ...
- 干货 | 列生成VRPTW子问题ESPPRC( Elementary shortest path problem with resource constraints)介绍附C++代码
00 前言 各位小伙伴大家好,相信大家已经看过前面column generation求解vehicle routing problems的过程详解.该问题中,子问题主要是找到一条reduced cos ...
- 【CF edu 27 G. Shortest Path Problem?】
time limit per test 3 seconds memory limit per test 512 megabytes input standard input output standa ...
- Codeforces 845G Shortest Path Problem?
http://codeforces.com/problemset/problem/845/G 从顶点1dfs全图,遇到环则增加一种备选方案,环上的环不需要走到前一个环上作为条件,因为走完第二个环可以从 ...
- AT [ABC177F] I hate Shortest Path Problem
因为每行只有一个区域不能往下走,因此我们可以来分析一下从起点到整个矩形每个位置的最短路.可以发现每一行的最短路只与上一行的最短路有关,假设我们知道上一行的最短路,上一行不能往下走的区间在 \([L, ...
- Solve Longest Path Problem in linear time
We know that the longest path problem for general case belongs to the NP-hard category, so there is ...
- Why longest path problem doesn't have optimal substructure?
We all know that the shortest path problem has optimal substructure. The reasoning is like below: Su ...
- 【CF938G】Shortest Path Queries(线段树分治,并查集,线性基)
[CF938G]Shortest Path Queries(线段树分治,并查集,线性基) 题面 CF 洛谷 题解 吼题啊. 对于每个边,我们用一个\(map\)维护它出现的时间, 发现询问单点,边的出 ...
随机推荐
- 51Nod 1009 数字1的个数 | 数位DP
题意: 小于等于n的所有数中1的出现次数 分析: 数位DP 预处理dp[i][j]存 从1~以j开头的i位数中有几个1,那么转移方程为: if(j == 1) dp[i][j] = dp[i-1][9 ...
- OScached页面缓存的入门使用
OSCache的使用: 一,环境的搭建: 1,把oscache.jar file放在 /WEB-INF/lib 目录下(Put the oscache.jar file in the /WEB-INF ...
- flask函数已定义参数却出现takes 0 positional arguments but 1 was given的问题
在flask中定义了一个简单的删除数据库内容的路由 测试却发现一直报错 说delete_history函数定义时没有接受参数,但是检查delete_history函数却发现没有问题 后来想了半天才发现 ...
- bzoj 2434 fail tree+dfs序
首先比较明显的是我们可以将字符串组建立ac自动机,那么对于询问s1字符串在s2字符串中出现的次数,就是在以s1结尾为根的fail tree中,子树有多少个节点是s2的节点,这样我们处理fail tre ...
- 转载:WebView
前言 现在很多App里都内置了Web网页(Hyprid App),比如说很多电商平台,淘宝.京东.聚划算等等,如下图 那么这种该如何实现呢?其实这是Android里一个叫WebView的组件实现的.今 ...
- 【shell】shell编程(六)-shell函数的应用
linux shell 可以用户定义函数,然后在shell脚本中可以随便调用. shell中函数的定义格式如下: [ function ] funname [()] { action; [return ...
- [NOI2008] [bzoj1061] 志愿者招募
还是一道费用流的题目.话不多说,进入正题. 题意:给定n个点和m种从l到r覆盖一层的费用,求满足所有点的覆盖层数都大等于权值的最小费用 分析:要做到区间修改,看似比较麻烦. 用差分把区间修改变成单点修 ...
- linux命令行todo列表管理工具Taskwarrior介绍
Taskwarrior 是一款在命令行下使用的TODO列表管理工具,或者说任务管理工具,灵活,快速,高效. 安装 在ubuntu 14.04 中,可从官方仓库安装task软件包 sudo apt-ge ...
- vim查找/替换字符串【转】
转自:http://www.cnblogs.com/GODYCA/archive/2013/02/22/2922840.html vi/vim 中可以使用 :s 命令来替换字符串.该命令有很多种不同细 ...
- gpio子系统和pinctrl子系统(中)
pinctrl子系统核心实现分析 pinctrl子系统的内容在drivers/pinctrl文件夹下,主要文件有(建议先看看pinctrl内核文档Documentation/pinctrl.txt): ...