http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1340

设x为环线的长度,要判断某个特定的x是否可行,不难将题目转为差分约束模型,用最短路求解,每个限制条件对应图中一条边,可行当且仅当图中没有负环。

如果x是整数,用最短路可以构造出整数解,因此现在需要对所有x同时判断,求出x的可行域,再求整数解个数。
由于边权是关于x的一次函数,且一次项系数绝对值<=1,可以用Bellman-Ford算法处理,此时一个点w到源点的距离dist(w)被表示为关于x的函数dist(w,x),且这个函数可以表示为一系列一次函数取min。最后判负环的时候相当于求 对每条长度为e(w,u)的边w->u都满足dist(w,x)+e(w,u)-dist(u,x)>=0 的 x的取值范围。
时间复杂度O(Tn^2m)。
#include<bits/stdc++.h>
typedef long long i64;
const i64 inf=1ll<<;
int T,n,m1,m2,ep;
i64 l[][];
void mins(i64&a,i64 b){if(a>b)a=b;}
i64 up(i64 A,i64 B){
if(B<)B=-B,A=-A;
i64 C=A/B;
return C+(C*B<A);
}
i64 dn(i64 A,i64 B){
if(B<)B=-B,A=-A;
i64 C=A/B;
return C-(C*B>A);
}
struct pos{
i64 x;
int y;
bool operator<(const pos&w)const{return x<w.x;}
}ps[];
int pp;
void ins(i64 L,i64 R){
ps[pp++]=(pos){L,};
ps[pp++]=(pos){R+,-};
}
struct ln{
i64 k,b,l,r;
i64 at(i64 x){
return k*x+b;
}
void chk(ln w){
i64 L=std::max(l,w.l),R=std::min(r,w.r);
if(L>R)return;
w.k=k-w.k;
w.b=b-w.b;
if(w.at(L)>=){
if(w.at(R)>=)ins(L,R);
else ins(L,dn(-w.b,w.k));
}else if(w.at(R)>=)ins(up(-w.b,w.k),R);
}
bool cmp(ln&w){
return at(l)>=w.at(l);
}
void cut(ln&w){
i64 K=k-w.k,B=b-w.b;
r=dn(-B,K);
w.l=r+;
}
}v1[],v2[];
void push(ln*a,int&ap,ln w){
for(;ap&&a[ap].cmp(w);--ap);
if(ap)a[ap].cut(w);
a[++ap]=w;
}
struct edge{
int x,y,a,b;
void upd(){
for(int i=;i<=n*;++i)if(i+b>=&&i+b<=n*)mins(l[y][i+b],l[x][i]+a);
}
void chk(){
int p1=,p2=;
for(int i=n*;i>=;--i){
if(l[x][i]<inf/)push(v1,p1,(ln){i-n+b,l[x][i]+a,n,inf});
if(l[y][i]<inf/)push(v2,p2,(ln){i-n,l[y][i],n,inf});
}
int i1=,i2=;
while(i1<=p1&&i2<=p2){
v1[i1].chk(v2[i2]);
if(v1[i1].r<=v2[i2].r)++i1;else ++i2;
}
}
}e[];
void ae(int a,int b,int d,int k){
e[ep++]=(edge){a,b,d,k};
}
int main(){
for(scanf("%d",&T);T;--T){
scanf("%d%d%d",&n,&m1,&m2);
ep=;
for(int i=;i<n;++i){
for(int j=;j<=n*;++j)l[i][j]=inf;
}
l[][n]=;
for(int i=;i<n;++i){
int a=i,b=(i+)%n,d=;
ae(b,a,-d,(a>b));
}
for(int i=,a,b,d;i<=m1;++i){
scanf("%d%d%d",&a,&b,&d);
ae(b,a,-d,(a>b));
}
for(int i=,a,b,d;i<=m2;++i){
scanf("%d%d%d",&a,&b,&d);
ae(a,b,d,-(a>b));
}
for(int t=;t<n;++t){
for(int i=;i<ep;++i)e[i].upd();
}
pp=;
for(int i=;i<ep;++i)e[i].chk();
std::sort(ps,ps+pp);
i64 ans=;
for(int i=,s=;i<pp;++i){
s+=ps[i].y;
if(s==ep)ans+=ps[i+].x-ps[i].x;
}
if(ans>inf/)ans=-;
printf("%lld\n",ans);
}
return ;
}

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

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

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

  2. 51nod1340地铁环线

    经典题. 经典差分约束模型. 但是 显然这个总长是有上下界的. 直接二分总长,判断有没有负环 如果没有负环好办,有负环就不知道怎么偏了. 因为没有单调性! (如果所有没有单调性的函数图像,都知道往哪里 ...

  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. React文档(十四)深入JSX

    根本上,JSX只是为React.createElement(component, props, ...children)函数提供语法糖.JSX代码是这样的: <MyButton color=&q ...

  2. 【CUDA】Win10 + VS2017新 CUDA 项目配置

    一.新建项目 打开VS2017 → 新建项目 → Win32控制台应用程序 → “空项目”打钩 二.调整配置管理器平台类型 右键项目 → 属性 → 配置管理器 → 全改为“x64” 三.配置生成属性 ...

  3. Java IO编程全解(二)——传统的BIO编程

    前面讲到:Java IO编程全解(一)——Java的I/O演进之路 网络编程的基本模型是Client/Server模型,也就是两个进程之间进行相互通信,其中服务端提供位置信息(绑定的IP地址和监听端口 ...

  4. keepalived高可用集群。

    keepalived故障切换转移原理1vrrp协议:(vritual router redundancy protocol)虚拟路由冗余协议,2故障转移.keepalived三大功能1实现物理高可用, ...

  5. springboot+thymeleaf+springbootJPA实现一个简单的增删改查

    1.springboot是什么,给我们带来了什么方便? 通过阅读springboot的书籍发现springboot最便利的是为我们自动配置了很多的东西,几乎可以实现无xml,甚至简单的无sql,为我们 ...

  6. jar 包启动

    java -Xms256m -Xmx512m -Xmn256m -jar /home/apps/video/video.jar --spring.profiles.active=test #!/bin ...

  7. Jsの练习-数组常用方法 -splice()

    splice() 功能:1.可以实现删除,插入和替换. 删除:可以删除任意数量的项,只需指定2个参数:要删除的第一项的位置和要删除的项数. 例如:splice(0,2)会删除数组中的前2项. 插入:可 ...

  8. Problem A: 平面上的点和线——Point类、Line类 (I)

    Description 在数学上,平面直角坐标系上的点用X轴和Y轴上的两个坐标值唯一确定,两点确定一条线段.现在我们封装一个“Point类”和“Line类”来实现平面上的点的操作. 根据“append ...

  9. Window服务器 安装 Memcached

    官网上并未提供 Memcached 的Window平台安装包,我们可以使用以下链接来下载,你需要根据自己的系统平台及需要的版本号点击对应的链接下载即可: 32位系统 1.2.5版本:http://st ...

  10. 群等变网络的pytorch实现

    CNN对于旋转不具有等变性,对于平移有等变性,data augmentation的提出就是为了解决这个问题,但是data augmentation需要很大的模型容量,更多的迭代次数才能够在训练数据集合 ...