经典题。

经典差分约束模型。

但是

显然这个总长是有上下界的。

直接二分总长,判断有没有负环

如果没有负环好办,有负环就不知道怎么偏了。

因为没有单调性

(如果所有没有单调性的函数图像,都知道往哪里走更优,

岂不是全都可以二分了

但是本题特殊在于,至少还是个区间!

二分左右端点。

负环记录k*mid+b的k,根据k的正负就可以知道哪个方向可能有解。

任意一个负环都可以判断的。

#include<bits/stdc++.h>
#define reg register int
#define il inline
#define fi first
#define se second
#define mk(a,b) make_pair(a,b)
#define numb (ch^'0')
#define pb push_back
#define solid const auto &
#define enter cout<<endl
#define pii pair<int,int>
using namespace std;
// #define int long long
typedef long long ll;
template<class T>il void rd(T &x){
char ch;x=;bool fl=false;while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);(fl==true)&&(x=-x);}
template<class T>il void output(T x){if(x/)output(x/);putchar(x%+'');}
template<class T>il void ot(T x){if(x<) putchar('-'),x=-x;output(x);putchar(' ');}
template<class T>il void prt(T a[],int st,int nd){for(reg i=st;i<=nd;++i) ot(a[i]);putchar('\n');}
namespace Modulo{
const int mod=;
il int ad(int x,int y){return x+y>=mod?x+y-mod:x+y;}
il int sub(int x,int y){return ad(x,mod-y);}
il int mul(int x,int y){return (ll)x*y%mod;}
il void inc(int &x,int y){x=ad(x,y);}
il void inc2(int &x,int y){x=mul(x,y);}
il int qm(int x,int y=mod-){int ret=;while(y){if(y&) ret=mul(x,ret);x=mul(x,x);y>>=;}return ret;}
template<class ...Args>il int ad(const int a,const int b,const Args &...args) {return ad(ad(a,b),args...);}
template<class ...Args>il int mul(const int a,const int b,const Args &...args) {return mul(mul(a,b),args...);}
}
// using namespace Modulo;
namespace Miracle{
const int N=;
const ll inf=1e9+;
int n,m1,m2;
struct node{
int fr,nxt,to;
int k,b;
ll val(ll x){
return (ll)k*x+b;
}
}e[N*N];
int hd[N],cnt;
void add(int x,int y,int k,int b){
// cout<<" add "<<x<<" to "<<y<<" k "<<k<<" b "<<b<<endl;
e[++cnt].nxt=hd[x];
e[cnt].to=y;e[cnt].k=k;e[cnt].b=b;
e[cnt].fr=x;
hd[x]=cnt;
}
ll dis[N],pre[N];
int has[N];
bool vis[N];
queue<int>q;
int spfa(ll mid){
memset(dis,0x3f,sizeof dis);
dis[]=;
memset(vis,,sizeof vis);
while(!q.empty()) q.pop();
has[]=;
q.push();
while(!q.empty()){
int x=q.front();q.pop();vis[x]=;
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
if(dis[y]>dis[x]+e[i].val(mid)){
dis[y]=dis[x]+e[i].val(mid);
pre[y]=i;
has[y]=has[x]+;
if(has[y]==n+){
int z=y;
memset(vis,,sizeof vis);
do{
vis[z]=;
z=e[pre[z]].fr;
}while(!vis[z]);
int k=;
int lp=z;
do{
k+=e[pre[z]].k;
z=e[pre[z]].fr;
}while(z!=lp); if(k>){
return ;
}else return -;
} if(!vis[y]){
vis[y]=;
q.push(y);
}
}
}
}
return ;
}
void clear(){
memset(hd,,sizeof hd);cnt=;
}
int main(){
// rd(n);rd(m);
int T;
rd(T);
while(T--){
clear();
rd(n);rd(m1);rd(m2);
for(reg i=;i<n;++i){
add(i,i-,,-);
}
add(,n-,,-);
int a,b,c;
for(reg i=;i<=m1;++i){
rd(a);rd(b);rd(c);
if(a<b){
add(b,a,,-c);
}else{
add(b,a,,-c);
}
}
for(reg i=;i<=m2;++i){
rd(a);rd(b);rd(c);
if(a<b){
add(a,b,,c);
}else{
add(a,b,-,c);
}
}
ll L=n,R=(ll)n*inf;
ll al=R+;
while(L<=R){
ll mid=(L+R)>>;
int lp=spfa(mid);
if(lp==-){
R=mid-;
}else if(lp==){
L=mid+;
}else{
al=mid;R=mid-;
}
}
L=n,R=(ll)n*inf;
ll ar=n-;
while(L<=R){
ll mid=(L+R)>>;
int lp=spfa(mid);
if(lp==-){
R=mid-;
}else if(lp==){
L=mid+;
}else{
ar=mid;L=mid+;
}
}
// cout<<" al "<<al<<" ar "<<ar<<endl;
if(ar<al){
puts("");
}else if(ar==(ll)n*inf){
puts("-1");
}else{
printf("%lld\n",ar-al+);
}
}
return ;
} }
signed main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
*/

二分条件:单调性

一切可以得知最优解方向的,也都可以二分

51nod1340地铁环线的更多相关文章

  1. 题解 [51nod1340]地铁环线

    题解 [51nod1340]地铁环线 题面 解析 本文参考这篇博客 一开始看到只有120行就打算写一写, 结果一刚就是三个星期摆摆摆 本来是当查分约束入门学的. step 1 首先来考虑下如果已知总长 ...

  2. 51nod1340 地铁环线

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1340 设x为环线的长度,要判断某个特定的x是否可行,不难将题目转为差分约 ...

  3. 【51nod 1340】地铁环线

    题目 有一个地铁环线,环线中有N个站台,标号为0,1,2,...,N-1.这个环线是单行线,一共由N条有向边构成,即从0到1,1到2,..k到k+1,...,N-2到N-1,N-1到0各有一条边.定义 ...

  4. [题解] 51 nod 1340 地铁环线

    不难看出这是一道差分约束的题目. 但是如果想按照通常的题目那样去建边的话,就会发现这句话--相邻两站的距离至少是1公里--建边后就直接让整个题出现了负环(默认是按求最短路建边),没法做了. 这时我们就 ...

  5. 【51nod】1340 地铁环线

    今天头非常疼,躲在家里没去机房 反正都要颓废了,然后花了一上午研究了一下这道神题怎么做-- 题解 首先我们发现,如果我们设\(dis[i]\)为从\(0\)节点走到\(i\)节点的距离 那么题目中给出 ...

  6. 【UR #2】跳蚤公路

    [UR #2]跳蚤公路 参照yjc方法.也就是地铁环线那个题. 求每个点不在负环内的x的取值范围.然后所有1到j能到i的j的范围取交.得到答案. 每个边形如kx+b的直线,每个环也是 每个点不在负环内 ...

  7. 【luogu P4005 清华集训2017】小Y和地铁

    题目描述 小 Y 是一个爱好旅行的 OIer.一天,她来到了一个新的城市.由于不熟悉那里的交通系统,她选择了坐地铁. 她发现每条地铁线路可以看成平面上的一条曲线,不同线路的交点处一定会设有 换乘站 . ...

  8. P4005 小 Y 和地铁

    题目描述 小 Y 是一个爱好旅行的 OIer.一天,她来到了一个新的城市.由于不熟悉那里的交通系统,她选择了坐地铁. 她发现每条地铁线路可以看成平面上的一条曲线,不同线路的交点处一定会设有 换乘站 . ...

  9. 华为上机测试题(地铁换乘-java)

    PS:自己写的,自测试OK,供大家参考. /* 高级题样题:地铁换乘描述:已知2条地铁线路,其中A为环线,B为东西向线路,线路都是双向的.经过的站点名分别如下,两条线交叉的换乘点用T1.T2表示.编写 ...

随机推荐

  1. LOJ 2721 「NOI2018」屠龙勇士——扩展中国剩余定理

    题目:https://loj.ac/problem/2721 1.注意别一输入 p[ i ] 就 a[ i ] %= p[ i ] ,因为在 multiset 里找的时候还需要真实值. 2.注意用 m ...

  2. LOJ 2979 「THUSCH 2017」换桌——多路增广费用流

    题目:https://loj.ac/problem/2979 原来的思路: 优化连边.一看就是同一个桌子相邻座位之间连边.相邻桌子对应座位之间连边. 每个座位向它所属的桌子连边.然后每个人建一个点,向 ...

  3. [CSP-S模拟测试]:联(小清新线段树)

    题目描述 由于出题人懒所以没有背景.一个无限长的$01$序列,初始全为$0$,每次选择一个区间$[l,r]$进行操作,有三种操作:$\bullet 1\ l\ r$将$[l,r]$中所有元素变成$1$ ...

  4. 北风设计模式课程---UML类图各符号含义

    北风设计模式课程---UML类图各符号含义 一.总结 一句话总结: 用脑子,挺好记的:实线关系肯定比虚线重,箭头.三角形.菱形的关系肯定依次加重,三角形是继承和实现, 1.UML类图中 线+箭头 表示 ...

  5. 建站手册-浏览器信息:Mozilla Firefox 浏览器

    ylbtech-建站手册-浏览器信息:Mozilla Firefox 浏览器 1.返回顶部 1. http://www.w3school.com.cn/browsers/browsers_firefo ...

  6. laravel框架基础知识总结

    一.laravel简介 laravel是一套优雅简介的PHP开发框架,受欢迎程度非常之高,功能强大,工具齐全:以下是本人在学习过程中记录的laravel比较基础的资料,权当学习笔记,请大神们多多指教 ...

  7. upc组队赛16 GCDLCM 【Pollard_Rho大数质因数分解】

    GCDLCM 题目链接 题目描述 In FZU ACM team, BroterJ and Silchen are good friends, and they often play some int ...

  8. python-django之cookie及session

    Cookie Cookie的由来 Http协议是无状态的 无状态的意思是每次都是独立的请求存在,它的执行情况和结果与前面的请求和后面的请求都无直接关系,它不会受到前面的请求响应情况直接影响,也不会直接 ...

  9. spring注解开发:容器中注册组件方式

    1.包扫描+组件标注注解 使用到的注解如下,主要针对自己写的类 @Controller @Service @Repository @Component @ComponentScan 参考 spring ...

  10. [fW]中断处理函数数组interrupt[]初始化

    中断处理函数数组interrupt[]初始化 2011-05-13 15:51:40 分类: LINUX 在系统初始化期间,trap_init()函数将对中断描述符表IDT进行第二次初始化(第一次只是 ...