[Usaco2015 Jan]Grass Cownoisseur Tarjan缩点+SPFA
考试的时候忘了缩点,人为dfs模拟缩点,没想到竟然跑了30分,RB爆发。。。
边是可以重复走的,所以在同一个强连通分量里,无论从那个点进入从哪个点出,所有的点一定能被一条路走到。 要使用缩点。
然后我们枚举每一条边,考虑如果将这条边反置的话,就是这条边的终点到1的点的权值(正向的)加上起点到1的点的权值(反向的);例:→→←→→
每个点到1的正向反向距离可以用两遍SPFA解决出来。
先使用tarjan缩点,记录每个点的权值,缩点后权值变为强连通分量中点的个数。缩完点之后重新建图,正向边存1,反向边存-1;
注意:在枚举每一条边求MAX时一定要判断它的起点和终点能否到达1。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<queue>
using namespace std;
#define pos(i,a,b) for(int i=(a);i<=(b);i++)
#define pos2(i,a,b) for(int i=(a);i>=(b);i--)
#define N 501000
int n,m;
struct haha
{
int next,to,w;
};
haha edgechu[N],edge[N];
int head[N],headchu[N],cnt=1,cntchu=1,cntt,hea;
int belong[N],low[N],dfn[N],ji=1,stack[N],instack[N];
int point[N];
void add(int u,int v,int w)
{
edge[cnt].to=v;
edge[cnt].next=head[u];
edge[cnt].w=w;
head[u]=cnt++;
}
void addchu(int u,int v,int w)
{
edgechu[cntchu].to=v;
edgechu[cntchu].next=headchu[u];
edgechu[cntchu].w=w;
headchu[u]=cntchu++;
}
void tarjan(int now)
{
low[now]=dfn[now]=ji;
ji++;
stack[++hea]=now;
instack[now]=1;
for(int v=headchu[now];v;v=edgechu[v].next)
{
int i=edgechu[v].to;
if(dfn[i]==-1)
{
tarjan(i);
low[now]=min(low[now],low[i]);
}
else
if(instack[i])
low[now]=min(low[now],dfn[i]);
}
if(low[now]==dfn[now])
{
cntt++;
int temp;
while(1)
{
temp=stack[hea--];
belong[temp]=cntt;
point[cntt]++;
instack[temp]=0;
//cout<<"temp="<<temp<<" cntt="<<cntt<<endl;
if(temp==now)
break;
}
}
}
int diszheng[N],disfan[N];
int flag[N];
void spfazheng(int x)
{
queue<int> q;
pos(i,1,n)
diszheng[i]=0;
memset(flag,0,sizeof(flag));
diszheng[x]=point[x];
q.push(x);
flag[x]=1;
int k;
while(!q.empty())
{
k=q.front();
for(int v=head[k];v;v=edge[v].next)
{
int i=edge[v].to;
if(edge[v].w>0&&diszheng[i]<diszheng[k]+point[i])
{
diszheng[i]=diszheng[k]+point[i];
if(!flag[i])
{
q.push(i);
flag[i]=1;
}
}
}
flag[q.front()]=0;
q.pop();
}
}
void spfafan(int x)
{
queue<int> q;
pos(i,1,n)
disfan[i]=0;
memset(flag,0,sizeof(flag));
disfan[x]=point[x];
q.push(x);
flag[x]=1;
int k;
while(!q.empty())
{
k=q.front();
for(int v=head[k];v;v=edge[v].next)
{
int i=edge[v].to;
if(edge[v].w<0&&disfan[i]<disfan[k]+point[i])
{
disfan[i]=disfan[k]+point[i];
if(!flag[i])
{
q.push(i);
flag[i]=1;
}
}
}
flag[q.front()]=0;
q.pop();
}
}
struct qian
{
int from,to;
}cun[N];
int road;
int ans;
int main()
{
//freopen("cown.in","r",stdin);
//freopen("cown.out","w",stdout);
scanf("%d%d",&n,&m);
memset(dfn,-1,sizeof(dfn));
pos(i,1,m)
{
int x,y;
scanf("%d%d",&x,&y);
addchu(x,y,1);
}
pos(i,1,n)
if(dfn[i]==-1)
tarjan(i);
pos(i,1,n)
for(int v=headchu[i];v;v=edgechu[v].next)
{
int j=edgechu[v].to;
if(belong[i]!=belong[j])
{
add(belong[i],belong[j],1);
add(belong[j],belong[i],-1);
cun[++road].from=belong[i];
cun[road].to=belong[j];
}
}
spfafan(belong[1]);spfazheng(belong[1]);
/*pos(i,1,cntt)
cout<<"diszheng[i]="<<diszheng[i]<<" disfan[i]="<<disfan[i]<<endl;*/
pos(i,1,road)
{
if(diszheng[cun[i].to]>0&&disfan[cun[i].from]>0)
ans=max(ans,diszheng[cun[i].to]+disfan[cun[i].from]-point[belong[1]]);
//cout<<"ans="<<ans<<endl;
}
printf("%d",ans);
//while(1);
return 0;
}
[Usaco2015 Jan]Grass Cownoisseur Tarjan缩点+SPFA的更多相关文章
- BZOJ 3887: [Usaco2015 Jan]Grass Cownoisseur tarjan + spfa
Code: #include <bits/stdc++.h> #define setIO(s) freopen(s".in","r",stdin) ...
- BZOJ3887 [Usaco2015 Jan] Grass Cownoisseur 【tarjan】【DP】*
BZOJ3887 [Usaco2015 Jan] Grass Cownoisseur Description In an effort to better manage the grazing pat ...
- [补档][Usaco2015 Jan]Grass Cownoisseur
[Usaco2015 Jan]Grass Cownoisseur 题目 给一个有向图,然后选一条路径起点终点都为1的路径出来,有一次机会可以沿某条边逆方向走,问最多有多少个点可以被经过? (一个点在路 ...
- bzoj3887: [Usaco2015 Jan]Grass Cownoisseur
题意: 给一个有向图,然后选一条路径起点终点都为1的路径出来,有一次机会可以沿某条边逆方向走,问最多有多少个点可以被经过?(一个点在路径中无论出现多少正整数次对答案的贡献均为1) =>有向图我们 ...
- BZOJ 3887/Luogu P3119: [Usaco2015 Jan]Grass Cownoisseur (强连通分量+最长路)
分层建图,反向边建在两层之间,两层内部分别建正向边,tarjan缩点后,拓扑排序求一次1所在强连通分量和1+n所在强联通分量的最长路(长度定义为路径上的强联通分量内部点数和).然后由于1所在强连通分量 ...
- [Usaco2015 Jan]Grass Cownoisseur 图论 tarjan spfa
先缩点,对于缩点后的DAG,正反跑spfa,枚举每条边进行翻转即可 #include<cstdio> #include<cstring> #include<iostrea ...
- BZOJ3887 [Usaco2015 Jan]Grass Cownoisseur[缩点]
首先看得出缩点的套路.跑出DAG之后,考虑怎么用逆行条件.首先可以不用,这样只能待原地不动.用的话,考虑在DAG上向后走,必须得逆行到1号点缩点后所在点的前面,才能再走回去. 于是统计从1号点缩点所在 ...
- 洛谷—— P3119 [USACO15JAN]草鉴定Grass Cownoisseur || BZOJ——T 3887: [Usaco2015 Jan]Grass Cownoisseur
http://www.lydsy.com/JudgeOnline/problem.php?id=3887|| https://www.luogu.org/problem/show?pid=3119 D ...
- [bzoj3887][Usaco2015 Jan]Grass Cownoisseur_trajan_拓扑排序_拓扑序dp
[Usaco2015 Jan]Grass Cownoisseur 题目大意:给一个有向图,然后选一条路径起点终点都为1的路径出来,有一次机会可以沿某条边逆方向走,问最多有多少个点可以被经过?(一个点在 ...
随机推荐
- [leetcode-556-Next Greater Element III]
Given a positive 32-bit integer n, you need to find the smallest 32-bit integer which has exactly th ...
- Linux配置LNMP环境(二)配置PHP
前言:本教程安装的PHP版本php-5.6.30(官方最后更新日期2017-01-19),教程编写日期2017-07-02.本教程中的下载地址是在写教程的时候从官方复制的,时间过长可能会有变化. 安装 ...
- (转载)Bonding技术指南
原文链接:http://www.wushiqin.com/?post=68 一.什么是网卡绑定及简单原理 网卡绑定也称作"网卡捆绑",就是使用多块物理网卡虚拟成为一块网卡,以提供负 ...
- 常见的排序算法总结(JavaScript)
引言 排序算法是数据结构和算法之中的基本功,无论是在笔试还是面试,还是实际运用中都有着很基础的地位.这不正直七月,每年校招的备战期,所以想把常见的排序算法记录下来.在本篇文章中的排序算法使用 Java ...
- ArrayList源码浅析(jdk1.8)
ArrayList的实质就是动态数组.所以可以通过下标准确的找到目标元素,因此查找的效率高.但是添加或删除元素会涉及到大量元素的位置移动,所以效率低. 一.构造方法 ArrayList提供了3个构造方 ...
- Luogu 2756 飞行员配对方案问题(二分图最大匹配)
Luogu 2756 飞行员配对方案问题(二分图最大匹配) Description 英国皇家空军从沦陷国征募了大量外籍飞行员.由皇家空军派出的每一架飞机都需要配备在航行技能和语言上能互相配合的2 名飞 ...
- VMware虚拟机下Ubuntu连不上网解决
虚拟机:VMware® Workstation 12 Pro Ubuntu: 14.04 问题描述:电脑从公司拿到家打开Ubuntu之后,打开自带Firefox浏览器,显示连接不上网,终端下ping ...
- Android 自定义帧动画
Android 自定义帧动画 Android L : Android Studio 帧动画 和gif图片类似,顺序播放准本好的图片文件:图片资源在xml文件中配置好 将图片按照预定的顺序一张张切换,即 ...
- shell统计文本中单词的出现次数
Ubuntu14.04 给定一个文本,统计其中单词出现的次数 方法1 # solution 1 grep与awk配合使用,写成一个sh脚本 fre.sh sh fre.sh wordfretest.t ...
- unrecognized font family "iconfont2"
起因:使用React Native开发App,需要使用自定义字体iconfont2.ttf,要在xCode中引入该字体 步骤:将字体文件拷贝到项目工程中,在Info.plist文件中添加Fonts p ...