题目传送门:https://agc011.contest.atcoder.jp/tasks/agc011_f

题目大意:

现有一条铁路,铁路分为\(1\sim n\)个区间和\(0\sim n\)个站台,区间\(i\)连接站台\(i-1\)和\(i\)

一列火车经过区间\(i\)会消耗\(A_i\),区间内的铁路是单向或者是双向的,现在你需要设计一个火车时间表,满足:

  • 所有火车从\(0\)到\(n\)或从\(n\)到\(0\)
  • 火车在区间中不得逗留
  • 两列同向的火车之间的时间间隔为\(K\)
  • 单向区间不得有两列相向而行的火车同时经过

输出某辆列车从\(0\)到\(n\)再返回的最小时间


我们记车\(0\)(终点为\(n\))在车站\(i\)等待了\(P_i\)分钟,记车\(1\)(终点\(0\))在车站\(i\)等待了\(-Q_i\)分钟,由于是在\(\%K\)意义下,所以这样是没有问题的

\(P_i\)和\(Q_i\)显然为非负整数,考虑对于某条单向区间\(x\),有:

车\(0\)在第\(x\)条铁路上的区间为\((\sum\limits_{i=1}^{x-1}A_i+\sum\limits_{i=0}^{x-1}P_i,\sum\limits_{i=1}^xA_i+\sum\limits_{i=0}^{x-1}P_i)\)

车\(1\)在第\(x\)条铁路上的区间为\((\sum\limits_{i=1}^{x-1}-A_i+\sum\limits_{i=0}^{x-1}-Q_i,\sum\limits_{i=1}^x-A_i+\sum\limits_{i=0}^{x-1}-Q_i)\)

显然这两个区间长度是一样的,然后由于这两个区间交集为空,于是我们可以转化为端点的不等式,解得\(\sum\limits_{i=0}^{x-1}P_i+Q_i\notin(-2\sum\limits_{i=1}^xA_i,-2\sum\limits_{i=1}^{x-1}A_i)\),我们设这段区间的补集为\([L_i,R_i]\),得到\(\sum\limits_{i=0}^{x-1}P_i+Q_i\in[L_i,R_i]\)

记\(S_x=\sum\limits_{i=0}^xP_i+Q_i\),答案即为\(S_n+2\sum\limits_{i=1}^{n}A_i\)问题转化为我们要最小化\(S_{n}\),且满足\(S_i\in[L_i,R_i]\),那么这题就成了:给定\(n\)个区间,任选起点,走\(n\)步,第\(i\)步需要落在区间\([L_i,R_i]\)中,求最小总路径长度

这显然是可以dp的,而且有个很显然的贪心结论:如果起点确定,那么每次走到下一个区间的左端点最优

于是我们设\(f_i\)表示当前在区间\([L_i,R_i]\),一直走到\(n\)的最短路径,那么每次转移时,我们需要找到一个\(j\)满足\(L_j\notin[L_i,R_i],i<j\),且\(j\)最小,转移即为\(f_i=f_j+dis(L_i,L_j)\),找\(j\)的过程是可以用线段树优化的,直接进行补集覆盖即可

最后枚举所有的起点得到最优解,加上\(\sum\limits_{i=1}^nA_i\)输出即可

/*program from Wolfycz*/
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define inf 0x7f7f7f7f
using namespace std;
typedef long long ll;
typedef unsigned int ui;
typedef unsigned long long ull;
inline char gc(){
static char buf[1000000],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,1000000,stdin),p1==p2)?EOF:*p1++;
}
inline int frd(){
int x=0,f=1; char ch=gc();
for (;ch<'0'||ch>'9';ch=gc()) if (ch=='-') f=-1;
for (;ch>='0'&&ch<='9';ch=gc()) x=(x<<3)+(x<<1)+ch-'0';
return x*f;
}
inline int read(){
int x=0,f=1; char ch=getchar();
for (;ch<'0'||ch>'9';ch=getchar()) if (ch=='-') f=-1;
for (;ch>='0'&&ch<='9';ch=getchar()) x=(x<<3)+(x<<1)+ch-'0';
return x*f;
}
inline void print(int x){
if (x<0) putchar('-'),x=-x;
if (x>9) print(x/10);
putchar(x%10+'0');
}
const int N=1e5;
struct S1{
#define ls (p<<1)
#define rs (p<<1|1)
int tree[(N<<3)+10];
void pushdown(int p){
if (!tree[p]) return;
tree[ls]=tree[rs]=tree[p];
tree[p]=0;
}
void Modify(int p,int l,int r,int x,int y,int v){
if (x>y) return;
if (x<=l&&r<=y){
tree[p]=v;
return;
}
pushdown(p);
int mid=(l+r)>>1;
if (x<=mid) Modify(ls,l,mid,x,y,v);
if (y>mid) Modify(rs,mid+1,r,x,y,v);
}
int Query(int p,int l,int r,int x){
if (l==r) return tree[p];
pushdown(p);
int mid=(l+r)>>1;
if (x<=mid) return Query(ls,l,mid,x);
else return Query(rs,mid+1,r,x);
}
#undef ls
#undef rs
}ST;//Segment Tree
int A[N+10],B[N+10];
int L[N+10],R[N+10];
ll sum[N+10],f[N+10];
int list[(N<<1)+10];
int n,K,T,tot;
ll Get(int x){
int tmp=ST.Query(1,1,T,x);
if (!tmp) return 0;
return f[tmp]+(list[L[tmp]]-list[x]+K)%K;
}
int main(){
n=read(),K=read();
bool flag=0;
for (int i=1;i<=n;i++){
A[i]=read(),B[i]=read();
sum[i]=sum[i-1]+A[i];
if (B[i]==2) continue;
if (2*A[i]>K) flag=1;
}
if (flag){
printf("-1\n");
return 0;
}
for (int i=1;i<=n;i++){
if (B[i]==1){
L[i]=(-2*sum[i-1]%K+K)%K;
R[i]=(-2*sum[i ]%K+K)%K;
}else L[i]=0,R[i]=K-1;
list[++tot]=L[i];
list[++tot]=R[i];
}
sort(list+1,list+1+tot);
T=unique(list+1,list+1+tot)-list-1;
for (int i=1;i<=n;i++){
L[i]=lower_bound(list+1,list+1+T,L[i])-list;
R[i]=lower_bound(list+1,list+1+T,R[i])-list;
}
for (int i=n;i;i--){
f[i]=Get(L[i]);
if (R[i]<L[i]) ST.Modify(1,1,T,R[i]+1,L[i]-1,i);
else ST.Modify(1,1,T,1,L[i]-1,i),ST.Modify(1,1,T,R[i]+1,T,i);
}
ll Ans=f[1];
for (int i=1;i<=T;i++) Ans=min(Ans,Get(i));
printf("%lld\n",Ans+2*sum[n]);
return 0;
}

AtCoder Grand Contest 011 F - Train Service Planning的更多相关文章

  1. 【Atcoder Grand Contest 011 F】Train Service Planning

    题意:给\(n+1\)个站\(0,\dots,n\),连续的两站\(i-1\)和\(i\)之间有一个距离\(A_i\),其是单行(\(B_i=1\))或双行(\(B_i=2\)),单行线不能同时有两辆 ...

  2. AtCoder Grand Contest 011

    AtCoder Grand Contest 011 upd:这篇咕了好久,前面几题是三周以前写的... AtCoder Grand Contest 011 A - Airport Bus 翻译 有\( ...

  3. AtCoder Grand Contest 002 F:Leftmost Ball

    题目传送门:https://agc002.contest.atcoder.jp/tasks/agc002_f 题目翻译 你有\(n*k\)个球,这些球一共有\(n\)种颜色,每种颜色有\(k\)个,然 ...

  4. AtCoder Grand Contest 017 F - Zigzag

    题目传送门:https://agc017.contest.atcoder.jp/tasks/agc017_f 题目大意: 找出\(m\)个长度为\(n\)的二进制数,定义两个二进制数的大小关系如下:若 ...

  5. AtCoder Grand Contest 003 F - Fraction of Fractal

    题目传送门:https://agc003.contest.atcoder.jp/tasks/agc003_f 题目大意: 给定一个\(H×W\)的黑白网格,保证黑格四连通且至少有一个黑格 定义分形如下 ...

  6. AtCoder Grand Contest 010 F - Tree Game

    题目传送门:https://agc010.contest.atcoder.jp/tasks/agc010_f 题目大意: 给定一棵树,每个节点上有\(a_i\)个石子,某个节点上有一个棋子,两人轮流操 ...

  7. AtCoder Grand Contest 016 F - Games on DAG

    题目传送门:https://agc016.contest.atcoder.jp/tasks/agc016_f 题目大意: 给定一个\(N\)点\(M\)边的DAG,\(x_i\)有边连向\(y_i\) ...

  8. Atcoder Grand Contest 038 F - Two Permutations(集合划分模型+最小割)

    洛谷题面传送门 & Atcoder 题面传送门 好久前做的题了--今天偶然想起来要补个题解 首先考虑排列 \(A_i\) 要么等于 \(i\),要么等于 \(P_i\) 这个条件有什么用.我们 ...

  9. Atcoder Grand Contest 015 F - Kenus the Ancient Greek(找性质+乱搞)

    洛谷题面传送门 & Atcoder 题面传送门 一道难度 Au 的 AGC F,虽然看过题解之后感觉并不复杂,但放在现场确实挺有挑战性的. 首先第一问很简单,只要每次尽量让"辗转相除 ...

随机推荐

  1. 如何动态地给vSphere虚拟机模板注入信息

    在做vSphere自动化安装过程中,遇到这样一个需求:将vCenter Server做成模板,在给用户自动化装好vSphere后, 下载vCenter Server模板并启动虚拟机,然后将vCente ...

  2. 01背包+卡精度 Hdu 2955

    <span style="color:#3333ff;">/* ---------------------------------------------------- ...

  3. 文本分析实例---QQ聊天记录分析

    对QQ聊天记录进行分析,由于每天产生的聊天记录比較多,所以选取的是从2月份整月的聊天记录数据.分析要产生的结果有三个,聊天记录中发消息的人前top15.统计24小时时间段那个时间段发贴人最多,还有对消 ...

  4. snip_进制转换代码段

    10进制转成16进制. /* 程序:10进制转16进制的C语言实现 描述: 关键: 获得余 获得整 整为零 则退出 */ #include<stdio.h> #define N 10 #d ...

  5. MYSQL强制使用索引和禁止使用索引

    mysql强制索引和禁止某个索引 1.mysql强制使用索引:force index(索引名或者主键PRI) 例如: select * from table force index(PRI) limi ...

  6. F08标准中Open命令的newunit选项

    从gfortran 4.5开始Open命令开始支持newunit选项,示例如下: integer :: u open(newunit=u, file="log.txt", posi ...

  7. 获取IOS应用安装列表

    原文转载至 http://blog.csdn.net/justinjing0612/article/details/8887747 转自鸟哥博客:http://blog.cnrainbird.com/ ...

  8. Kafka在Windows安装运行

    摘要:本文主要说明了如何在Windows安装运行Kafka 一.安装JDK 过程比较简单,这里不做说明. 最后打开cmd输入如下内容,表示安装成功 二.安装zooeleeper 下载安装包:http: ...

  9. HDU 3714/UVA1476 Error Curves

    Error Curves Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Tota ...

  10. SDIO卡 了解

    SDIO接口是在SD接口基础上发展起来的,SDIO接口兼容SD接口.SDIO协议又在SD卡协议之上添加了CMD52(一般用来访问寄存器)和CMD53(字节和块传输)命令.SDIO和SD卡规范间的一个重 ...