题目链接

POJ2374

题解

题意:

给出\(n\)个平行于\(x\)轴的栅栏,求从一侧栅栏的某个位置出发,绕过所有栅栏到达另一侧\(x = 0\)位置的最短水平距离

往上说都是线段树优化dp

我写了一个奇怪的线段树过了,似乎并没有和dp沾边

因为每次都是从某个栅栏的端点出发,到达某个位置的值等于[所有这些可出发的端点已产生的代价 + 到达这个点的距离] 的最小值

就用线段树维护每个区间

\(lm[u]\)表示在这个区间内的端点到达左端点的最小代价

\(rm[u]\)表示到右端点的最小代价

然后每出现一层栅栏,计算出左右端点来更新线段树,同时清空被新栅栏覆盖的区间【因为这些端点已经无法直接到达下一层区间了】

PS:

一开始看错题了,栅栏的顺序是反过来的

在察觉看错题之前,拿了往上若干AC程序对拍

拍出一个惊人小数据,竟所有程序输出都不一样!!

5 0

3 4

-4 5

-4 0

1 3

2 3

最终画图钦定,,我把它们hack掉了

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define LL long long int
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define cls(s) memset(s,0,sizeof(s))
#define ls (u << 1)
#define rs (u << 1 | 1)
using namespace std;
const int maxn = 200005,maxm = 100005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
const int ll = 1,rr = 200001,P = 100001;
int lm[maxn << 2],rm[maxn << 2],tag[maxn << 2];
void upd(int u,int l,int r){
int mid = l + r >> 1;
lm[u] = min(lm[ls],lm[rs] + (mid - l + 1));
rm[u] = min(rm[rs],rm[ls] + (r - mid));
}
void pd(int u){
if (tag[u]){
lm[ls] = rm[ls] = lm[rs] = rm[rs] = INF;
tag[ls] = tag[rs] = 1;
tag[u] = 0;
}
}
void modify(int u,int l,int r,int pos,int v){
if (l == r){
lm[u] = min(lm[u],v);
rm[u] = min(rm[u],v);
return;
}
pd(u);
int mid = l + r >> 1;
if (mid >= pos) modify(ls,l,mid,pos,v);
else modify(rs,mid + 1,r,pos,v);
upd(u,l,r);
}
void reset(int u,int l,int r,int L,int R){
if (l >= L && r <= R){lm[u] = rm[u] = INF; tag[u] = 1; return;}
pd(u);
int mid = l + r >> 1;
if (mid >= L) reset(ls,l,mid,L,R);
if (mid < R) reset(rs,mid + 1,r,L,R);
upd(u,l,r);
}
int query(int u,int l,int r,int pos){
if (l == r) return min(lm[u],rm[u]);
pd(u);
int mid = l + r >> 1;
if (mid >= pos) return min(lm[rs] + (mid - pos + 1),query(ls,l,mid,pos));
return min(rm[ls] + (pos - mid),query(rs,mid + 1,r,pos));
}
void build(int u,int l,int r){
if (l == r){lm[u] = rm[u] = INF; return;}
int mid = l + r >> 1;
build(ls,l,mid);
build(rs,mid + 1,r);
upd(u,l,r);
}
int ql[maxn],qr[maxn];
int main(){
int n,s,l,r,ld,rd;
while (~scanf("%d%d",&n,&s)){
s += P;
build(1,ll,rr);
modify(1,ll,rr,s,0);
for (int i = 1; i <= n; i++) ql[i] = read(),qr[i] = read();
for (int i = n; i; i--){
l = ql[i] + P; r = qr[i] + P;
ld = query(1,ll,rr,l);
rd = query(1,ll,rr,r);
if (l + 1 <= r - 1) reset(1,ll,rr,l + 1,r - 1);
modify(1,ll,rr,l,ld);
modify(1,ll,rr,r,rd);
}
printf("%d\n",query(1,ll,rr,P));
}
return 0;
}

POJ2374 Fence Obstacle Course 【线段树】的更多相关文章

  1. poj2374 Fence Obstacle Course[线段树+DP]

    https://vjudge.net/problem/POJ-2374 吐槽.在这题上面磕了许久..英文不好题面读错了qwq,写了个错的算法搞了很久..A掉之后瞥了一眼众多julao题解,**,怎么想 ...

  2. POJ 2374 Fence Obstacle Course(线段树+动态规划)

    Fence Obstacle Course Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 2524   Accepted:  ...

  3. POJ2374 Fence Obstacle Course

    题意 Language:Default Fence Obstacle Course Time Limit: 3000MS Memory Limit: 65536K Total Submissions: ...

  4. 【BZOJ3387】[Usaco2004 Dec]Fence Obstacle Course栅栏行动 线段树

    [BZOJ3387][Usaco2004 Dec]Fence Obstacle Course栅栏行动 Description 约翰建造了N(1≤N≤50000)个栅栏来与牛同乐.第i个栅栏的z坐标为[ ...

  5. (困难) CF 484E Sign on Fence,整体二分+线段树

    Bizon the Champion has recently finished painting his wood fence. The fence consists of a sequence o ...

  6. Gym 101911E "Painting the Fence"(线段树区间更新+双端队列)

    传送门 题意: 庭院中有 n 个围栏,每个围栏上都被涂上了不同的颜色(数字表示): 有 m 条指令,每条指令给出一个整数 x ,你要做的就是将区间[ x第一次出现的位置 , x最后出现的位置 ]中的围 ...

  7. 「CF484E」Sign on Fence「整体二分」「线段树」

    题意 给定一个长度为\(n\)的正整数序列,第\(i\)个数为\(h_i\),\(m\)个询问,每次询问\((l, r, w)\),为\([l, r]\)所有长度为\(w\)的子区间最小值的最大值.( ...

  8. Fence Obstacle Course

    Fence Obstacle Course 有n个区间自下而上有顺序的排列,标号\(1\sim n\),第i个区间记做\([l_i,r_i]\),现在从第n个区间的起点s出发(显然s在\([l_n,r ...

  9. 【Codeforces720D】Slalom 线段树 + 扫描线 (优化DP)

    D. Slalom time limit per test:2 seconds memory limit per test:256 megabytes input:standard input out ...

随机推荐

  1. idea中使用逆向工程----三部曲

    逆向工程小伙伴可能都知道,可以根据公司大佬的数据库简单创建实体类和dao接口以及mapper的映射文件,逆向工程可能在数据库字段比较少的时候体现不会方便,但是当参与到数据库字段比较多的时候,我们不可能 ...

  2. Hutool Wiki For java

    发现一款不错的java工具类, http://www.hutool.cn/ Hutool的使用文档. 项目见 https://github.com/looly/hutool 以及 http://git ...

  3. Python学习笔记:PEP8常用编程规范

    PEP8编码规范是一种非常优秀的编码规范,也得到了Python程序员的普遍认可,如果实践中或者项目中没有统一的编码规范,建议尽量遵循PEP8编码规范,当然如果项目中已经有了自身的编码规范,应当优先遵循 ...

  4. Pythony的数据类型和变量使用方法详解

    数据类型:计算机顾名思义就是可以做数学计算的机器,因此,计算机程序理所当然地可以处理各种数值.但是,计算机能处理的远不止数值,还可以处理文本.图形.音频.视频.网页等各种各样的数据,不同的数据,需要定 ...

  5. POJ:2674-Linear world(名字交换碰撞)

    Linear world Time Limit: 3000MS Memory Limit: 65536K Total Submissions: 4514 Accepted: 1025 Descript ...

  6. [Luogu1341]无序字母对(欧拉回路)

    按题意给定字符串建无向图,找欧拉回路 按照定义,当没有奇数度点或者只有2个奇数度点时才有欧拉回路 Code #include <cstdio> #include <algorithm ...

  7. python-12正则表达式

    import re #re.search方法 re.search 扫描整个字符串并返回第一个成功的匹配. re.match('com', 'www.runoob.com') #匹配失败 None re ...

  8. python文件,字符串,二进制的读写

    读文件: f = open('/Users/michael/test.txt', 'r') #一次读取文件的全部内容 f.read() #文件使用完毕后必须关闭,因为文件对象会占用操作系统的资源,并且 ...

  9. TouTiao开源项目 分析笔记1

    1.InitApp==>项目的入口Application 1.1.继承了MultiDexApplication 超过65K方法的APP,会遇到65535的错误.原因就是为了支持比较大型的APP而 ...

  10. 原码、反码、补码、BCD码、格雷码

    二进制的最高位表示这个二进制的正负符号(0为正,1为负),其余各位数表示其数值本身称为原码. 正数的反码等于原码,负数的反码是在原码的基础上,符号位不变,其余各位取反. 正数的补码等于原码,负数的补码 ...