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. 【微信小程序开发】使用button标签的open-type="getUserInfo"引导用户去授权

    一. 前言 小程序官方文档,上面说明 > wx.getUserInfo(OBJECT) 注意:此接口有调整,使用该接口将不再出现授权弹窗,请使用 <button open-type=&qu ...

  2. 『TensorFlow』命令行参数解析

    argparse很强大,但是我们未必需要使用这么繁杂的东西,TensorFlow自己封装了一个简化版本的解析方式,实际上是对argparse的封装 脚本化调用tensorflow的标准范式: impo ...

  3. python -- 面向对象 - 反射

    1.isinstance ,type, issubclass       isinstance:判断给的对象是否是**类型       type:返回**对象的数据类型       issubclas ...

  4. Spring Boot 2.0尝鲜-动态 Banner

    配置依赖 使用 Spring Boot 2.0 首先需要将项目依赖包替换为刚刚发布的 2.0 RELEASE,现在网站https://start.spring.io/也将 Spring Boot 2. ...

  5. css颜色的五种表示方法

    一.最简单.最古老的颜色类型在CSS颜色的关键词,如red blue等. 二.十六进制值,如#0000. 三.RGB: rgb(255,0,0),这是给定的三个参数表示的红色,绿色和蓝色通道的颜色值每 ...

  6. 使用matlab生成用于ROM初始化的coe文件(转)

    reference:https://www.cnblogs.com/chensimin1990/p/9759368.html t=0:2*pi/2^12:2*pi; y=0.5*sin(t)+0.5; ...

  7. 第一个java程序中文乱码以及如何解决

    出现问题:编码gbk的不可映射字段 原因:.java文件的编码与cmd命令执行器使用的编码不一致 我们使用的.java文件的编码为UTF-8 Cmd默认使用的编码为GBK: 解决方式统一编码: 方法1 ...

  8. java集合之List。

    实际上有两种List:一种是基本的ArrayList其优点在于随机访问元素,另一种是更强大的LinkedList它并不是为快速随机访问设计的,而是具有一套更通用的方法. List:次序是List最重要 ...

  9. Hadoop学习笔记01_Hadoop搭建

    想往大数据方向转, 难度肯定是有的. 基础知识肯定是要有的,如果是熟悉JAVA开发的人,转向应该优势大. 像我这样的,只有Linux基础以及简单的PHP基础的人,转向难度很大.但是事在人为,努力学习多 ...

  10. C#磁性窗体设计

    using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using Sy ...