题面

Pinball的游戏界面由m+2行、n列组成。第一行在顶端。一个球会从第一行的某一列出发,开始垂直下落,界面上有一些漏斗,一共有m个漏斗分别放在第2m+1行,第i个漏斗的作用是把经过第i+1行且列数在AiBi之间的球,将其移到下一行的第Ci列。 使用第i个漏斗需要支付Di的价钱,你需要保留一些漏斗使得球无论从第一行的哪一列开始放,都只可能到达第m+2行的唯一 一列,求花费的最少代价。

(样例的图)

(我们保留2,4,5即可,代价为5+3+12=20)

Input

第一行两个数,m和n。m<=100000,2<=n<=1000000000

接下来m行,第i+1行描述第i个漏斗的属性,Ai,Bi,Ci,Di (1<=Ai<=Ci<=Bi<=n, 1<=Di<=1000000000)。

Output

若不存在一种方案能满足条件则输出-1,否则输出最小花费

Input示例

5 6

3 5 4 8

1 4 3 5

4 6 5 7

5 6 5 3

3 5 4 12

Output示例

20

题解

首先发现,对于一个漏斗来说,不管它上面的怎么选,它能贡献的肯定是一个区间

那么只要能够到\(1\),又能够到\(n\),那么就可以了

我们枚举最下面的漏斗是哪个,那么不难发现它需要两个满足\(C_i\)在\(A_i\)和\(B_i\)之间的漏斗,且一个能使它够到左边,一个能使它够到右边

简单来说,就是设\(lmn_i\)为必选第\(i\)个漏斗,且范围包含\([1,C_i]\)时的最小代价,\(rmn_i\)为必选第\(i\)个漏斗,且范围包含\([C_i,n]\)时的最小代价

然后我们枚举最下面的漏斗,答案即为满足\(A_i\leq C_j\leq B_i\)的最小的\(lmn_j\)和最小的\(rmn_j\)(两个不是同一个)

那么用线段树维护就好了

感觉比较难讲清楚,看代码比较好

//minamoto
#include<bits/stdc++.h>
#define R register
#define ll long long
#define ls (p<<1)
#define rs (p<<1|1)
#define inf 0x3f3f3f3f3f3f3f3f
#define fp(i,a,b) for(R int i=a,I=b+1;i<I;++i)
#define fd(i,a,b) for(R int i=a,I=b-1;i>I;--i)
#define go(u) for(int i=head[u],v=e[i].v;i;i=e[i].nx,v=e[i].v)
template<class T>inline bool cmax(T&a,const T&b){return a<b?a=b,1:0;}
template<class T>inline bool cmin(T&a,const T&b){return a>b?a=b,1:0;}
using namespace std;
char buf[1<<21],*p1=buf,*p2=buf;
inline char getc(){return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++;}
int read(){
R int res=1,f=1;R char ch;
while((ch=getc())>'9'||ch<'0')(ch=='-')&&(f=-1);
for(res=ch-'0';(ch=getc())>='0'&&ch<='9';res=res*10+ch-'0');
return res*f;
}
const int N=1e5+5;
int l[N],r[N],c[N],d[N],b[N];
ll lmn[N<<2],rmn[N<<2],lx,rx,res;
int n,m,lim;
void update(int p,int l,int r,int x){
cmin(lmn[p],lx),cmin(rmn[p],rx);
if(l==r)return;
int mid=(l+r)>>1;
x<=mid?update(ls,l,mid,x):update(rs,mid+1,r,x);
}
void query(int p,int l,int r,int ql,int qr){
if(ql<=l&&qr>=r)return cmin(lx,lmn[p]),cmin(rx,rmn[p]),void();
int mid=(l+r)>>1;
if(ql<=mid)query(ls,l,mid,ql,qr);
if(qr>mid)query(rs,mid+1,r,ql,qr);
}
int main(){
freopen("pinball.in","r",stdin);
freopen("pinball.out","w",stdout);
n=read(),m=read();
fp(i,1,n)l[i]=read(),r[i]=read(),c[i]=b[i]=read(),d[i]=read();
sort(b+1,b+1+n),lim=unique(b+1,b+1+n)-b-1;
memset(lmn,0x3f,sizeof(lmn));
memset(rmn,0x3f,sizeof(rmn));
res=inf;
fp(i,1,n){
lx=(l[i]==1)?0:inf;
rx=(r[i]==m)?0:inf;
l[i]=lower_bound(b+1,b+1+lim,l[i])-b;
r[i]=upper_bound(b+1,b+1+lim,r[i])-b-1;
c[i]=lower_bound(b+1,b+1+lim,c[i])-b;
query(1,1,lim,l[i],r[i]);
cmin(res,lx+rx+d[i]);
lx+=d[i],rx+=d[i];
update(1,1,lim,c[i]);
}
printf("%lld\n",res==inf?-1:res);
return 0;
}

51nod 1781 Pinball(线段树)的更多相关文章

  1. 51nod 1272 思维/线段树

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1272 1272 最大距离 题目来源: Codility 基准时间限制:1 ...

  2. 51nod 1376【线段树维护区间最大值】

    引自:wonter巨巨的博客 定义 dp[i] := 以数字 i(不是下标 i)为结尾的最长上升长度 然后用线段树维护 dp[i]: 每个节点维护 2 个信息,一个是当前区间的最大上升长度,一个是最大 ...

  3. 51Nod 欢乐手速场1 A Pinball[DP 线段树]

    Pinball xfause (命题人)   基准时间限制:1 秒 空间限制:262144 KB 分值: 20 Pinball的游戏界面由m+2行.n列组成.第一行在顶端.一个球会从第一行的某一列出发 ...

  4. 51nod 1593 公园晨跑 | ST表(线段树?)思维题

    51nod 1593 公园晨跑 有一只猴子,他生活在一个环形的公园里.有n棵树围绕着公园.第i棵树和第i+1棵树之间的距离是 di ,而第n棵树和第一棵树之间的距离是 dn .第i棵树的高度是 hi ...

  5. 51nod 1364 最大字典序排列(线段树)

    1364 最大字典序排列基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 给出一个1至N的排列,允许你做不超过K次操作,每次操作可以将相邻的两个数交换,问能够得到的字 ...

  6. 51nod 1376 最长递增子序列的数量(线段树)

    51nod 1376 最长递增子序列的数量 数组A包含N个整数(可能包含相同的值).设S为A的子序列且S中的元素是递增的,则S为A的递增子序列.如果S的长度是所有递增子序列中最长的,则称S为A的最长递 ...

  7. 51nod 1199 Money out of Thin Air(线段树+树剖分)

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1199 题意: 思路:因为是一棵树,所以需要把它剖分一下再映射到线段树上, ...

  8. 51nod 1463 找朋友(线段树+离线处理)

    http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1463 题意: 思路: 好题! 先对所有查询进行离线处理,按照右区间排序, ...

  9. 51nod 1766 树上的最远点对 | LCA ST表 线段树 树的直径

    51nod 1766 树上的最远点对 | LCA ST表 线段树 树的直径 题面 n个点被n-1条边连接成了一颗树,给出a~b和c~d两个区间,表示点的标号请你求出两个区间内各选一点之间的最大距离,即 ...

随机推荐

  1. python 正则表达式(二)

    下面列举了Python3的所有符号用法,别背,千万别背,用到时来查就行. 字符 含义 . 表示匹配除了换行符外的任何字符注:通过设置 re.DOTALL 标志可以使 . 匹配任何字符(包含换行符) | ...

  2. Python基础-读取excel

    import xlrdbook = xlrd.open_workbook('lanxia.xls')sheet = book.sheet_by_name('sheet1')rows = sheet.n ...

  3. Oracle_Exception_01_The Network Adapter could not establish the connection

    1.IP错误或端口错误. 在设置URL时错误,例如 url="jdbc:oracle:thin:@192.168.1.11:1521:mas" 数据库服务器不正确:ping 服务器 ...

  4. QWidget、QMainWindow、QFrame、QWindow、QDialog、QScrollArea区别

    QWidget是所有可视化控件的基类,可以直接渲染出一个窗口来. QMainWindow用来表示一个主窗口,这个主窗口可以设置菜单和工具栏 QFrame用来表示一个框架,用来当作容器,因为可以设置它的 ...

  5. leetcode 7 Reverse Integer(水题)

    so easy,注意一下输入不爆int但是反转以后可能爆int. class Solution { public: int gao(int w){ ) ; else{ ; while(w--){ an ...

  6. linux命令学习笔记(0):man命令

    Linux提供了丰富的帮助手册,当你需要查看某个命令的参数时不必到处上网查找,只要man一下即可. Linux的man手册共有以下几个章节: 代號 代表內容 使用者在shell中可以操作的指令或可执行 ...

  7. mci播放mp3

    1MIDI的播放---- 乐器数字化接口(MIDI)是由音乐界的一些大公司(包括生产电子音乐合成器的公司)制订的一项协议,后来被计算机产业所采用并成为多媒体音乐文件的标准格式.MIDI文件一般较小,对 ...

  8. BZOJ3680:吊打XXX

    我对模拟退火的理解:https://www.cnblogs.com/AKMer/p/9580982.html 我对爬山的理解:https://www.cnblogs.com/AKMer/p/95552 ...

  9. python 3中使用getattr和*args时, 出现传入参数不一致的问题

    今天在用python3的getattr时遇到一个问题, 就是老提示传入参数和函数前面不一致, 代码为: class Test:      def __init__(self, name):       ...

  10. Excel对重复数据分组,求出不同的数据(office 2013)

    第一步: 第二步: 第三步: