multiset || 线段树 HDOJ 4302 Holedox Eating
题意:一个长度L的管子,起点在0。n次操作,0 p表示在p的位置放上蛋糕,1表示去吃掉最近的蛋糕(如果左右都有蛋糕且距离相同,那么吃同方向的蛋糕),问最终走了多少路程
分析:用multiset来保存蛋糕的位置,以当前的位置进行二分查找相邻的蛋糕的位置,模拟这个过程。当然也可以用线段树单点更新维护。
收获:当结果和暴力程序跑出来的一样却WA时,想想是否遗漏了某些情况的讨论
multiset:
/************************************************
* Author :Running_Time
* Created Time :2015-8-22 9:31:50
* File Name :C.cpp
************************************************/ #include <cstdio>
#include <algorithm>
#include <iostream>
#include <sstream>
#include <cstring>
#include <cmath>
#include <string>
#include <vector>
#include <queue>
#include <deque>
#include <stack>
#include <list>
#include <map>
#include <set>
#include <bitset>
#include <cstdlib>
#include <ctime>
using namespace std; #define lson l, mid, rt << 1
#define rson mid + 1, r, rt << 1 | 1
typedef long long ll;
const int N = 1e5 + 10;
const int INF = 0x3f3f3f3f;
const int MOD = 1e9 + 7; int main(void) {
int T, cas = 0; scanf ("%d", &T);
while (T--) {
multiset<int> S; S.clear ();
multiset<int>::iterator it, l, r;
int n, m; scanf ("%d%d", &n, &m);
ll ans = 0; int now = 0, d = 1; //d == 0:左 d == 1: 右
while (m--) {
int op, p; scanf ("%d", &op);
if (op == 0) {
scanf ("%d", &p); S.insert (p);
}
else {
if (S.empty ()) continue; //没蛋糕了
it = S.lower_bound (now); r = it, l = it;
if (it == S.end ()) { //右边没蛋糕
l--;
ans += (now - *l);
if (*l < now) d = 0; //当蛋糕在左边时才改变方向,下面类似
now = *l;
S.erase (l);
}
else if (it == S.begin ()) { //左边没蛋糕
ans += (*r - now);
if (now < *r) d = 1;
now = *r;
S.erase (r);
}
else { //左右都有蛋糕
l--;
int dt0 = now - *l, dt1 = *r - now;
if (dt0 == dt1) { //左右距离相等
if (d == 0) {
ans += dt0; now = *l;
S.erase (l);
}
else {
ans += dt1; now = *r;
S.erase (r);
}
}
else if (dt0 < dt1) { //吃左边的蛋糕
ans += dt0; now = *l;
if (dt0) d = 0;
S.erase (l);
}
else { //吃右边的蛋糕
ans += dt1; now = *r;
if (dt1) d = 1;
S.erase (r);
}
}
}
} printf ("Case %d: %I64d\n", ++cas, ans);
} return 0;
}
Segment_Tree:
#include <bits/stdc++.h>
using namespace std; #define lson l, mid, o << 1
#define rson mid + 1, r, o << 1 | 1
typedef long long ll;
const int N = 1e5 + 5;
const int INF = 0x3f3f3f3f;
struct Segment_Tree {
struct Node {
int mx, mn, cnt;
}node[N<<2];
void push_up(int o) {
node[o].mx = max (node[o<<1].mx, node[o<<1|1].mx);
node[o].mn = min (node[o<<1].mn, node[o<<1|1].mn);
}
void build(int l, int r, int o) {
if (l == r) {
node[o].mx = -INF; node[o].mn = INF;
node[o].cnt = 0; return ;
}
int mid = l + r >> 1;
build (lson); build (rson);
push_up (o);
}
void updata(int p, int c, int l, int r, int o) {
if (l == r && p == l) {
node[o].mx = node[o].mn = p;
node[o].cnt += c;
if (!node[o].cnt) {
node[o].mx = -INF; node[o].mn = INF;
}
return ;
}
int mid = l + r >> 1;
if (p <= mid) updata (p, c, lson);
else updata (p, c, rson);
push_up (o);
}
int query_left(int ql, int qr, int l, int r, int o) {
if (ql <= l && r <= qr) {
return node[o].mx;
}
int mid = l + r >> 1, ret = -INF;
if (ql <= mid) ret = max (ret, query_left (ql, qr, lson));
if (qr > mid) ret = max (ret, query_left (ql, qr, rson));
return ret;
}
int query_right(int ql, int qr, int l, int r, int o) {
if (ql <= l && r <= qr) {
return node[o].mn;
}
int mid = l + r >> 1, ret = INF;
if (ql <= mid) ret = min (ret, query_right (ql, qr, lson));
if (qr > mid) ret = min (ret, query_right (ql, qr, rson));
return ret;
}
}st; int main(void) {
int T, cas = 0; scanf ("%d", &T);
while (T--) {
int n, m; scanf ("%d%d", &n, &m);
n++;
st.build (1, n, 1);
ll ans = 0; int now = 1, d = 1;
while (m--) {
int op, p; scanf ("%d", &op);
if (op == 0) {
scanf ("%d", &p); p++;
st.updata (p, 1, 1, n, 1);
}
else {
int p1 = st.query_left (0, now, 1, n, 1);
int p2 = st.query_right (now, n, 1, n, 1);
if (p1 == -INF && p2 == INF) continue;
if (p2 == INF) {
ans += now - p1;
if (p1 < now) d = 0;
now = p1;
st.updata (p1, -1, 1, n, 1);
}
else if (p1 == -INF) {
ans += p2 - now;
if (p2 > now) d = 1;
now = p2;
st.updata (p2, -1, 1, n, 1);
}
else {
int dt0 = now - p1, dt1 = p2 - now;
if (dt0 == dt1) {
if (d == 0) {
ans += dt0; now = p1;
st.updata (p1, -1, 1, n, 1);
}
else {
ans += dt1; now = p2;
st.updata (p2, -1, 1, n, 1);
}
}
else if (dt0 < dt1) {
ans += dt0; now = p1;
if (dt0) d = 0;
st.updata (p1, -1, 1, n, 1);
}
else {
ans += dt1; now = p2;
if (dt1) d = 1;
st.updata (p2, -1, 1, n, 1);
}
}
}
}
printf ("Case %d: %I64d\n", ++cas, ans);
} return 0;
}
multiset || 线段树 HDOJ 4302 Holedox Eating的更多相关文章
- HDU 4302 Holedox Eating (线段树模拟)
题意:一个老鼠在一条长度为L的直线上跑,吃蛋糕,老鼠只能沿直线移动.开始时没有蛋糕,老鼠的初始位置是0. 有两个操作,0 x 代表在位置x添加一个蛋糕: 1 代表老鼠想吃蛋糕.老鼠每次都会选择离自己最 ...
- hdu 4302 Holedox Eating(优先队列/线段树)
题意:一只蚂蚁位与原点,在x轴正半轴上会不时地出现一些蛋糕,蚂蚁每次想吃蛋糕时选取最近的去吃,如果前后距离相同,则吃眼前的那一块(即方向为蚂蚁的正前),求最后蚂蚁行进距离. 思路:优先队列q存储蚂蚁前 ...
- 并查集+树链剖分+线段树 HDOJ 5458 Stability(稳定性)
题目链接 题意: 有n个点m条边的无向图,有环还有重边,a到b的稳定性的定义是有多少条边,单独删去会使a和b不连通.有两种操作: 1. 删去a到b的一条边 2. 询问a到b的稳定性 思路: 首先删边考 ...
- EC R 87 div2 D. Multiset 线段树 树状数组 二分
LINK:Multiset 主要点一下 二分和树状数组找第k大的做法. 线段树的做法是平凡的 开一个数组实现就能卡过. 考虑如树状数组何找第k大 二分+查询来判定是不优秀的. 考虑树状数组上倍增来做. ...
- 树链剖分+线段树 HDOJ 4897 Little Devil I(小恶魔)
题目链接 题意: 给定一棵树,每条边有黑白两种颜色,初始都是白色,现在有三种操作: 1 u v:u到v路径(最短)上的边都取成相反的颜色 2 u v:u到v路径上相邻的边都取成相反的颜色(相邻即仅有一 ...
- HDU 4302 Holedox Eating (STL + 模拟)
Holedox Eating Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...
- 树状数组+二分||线段树 HDOJ 5493 Queue
题目传送门 题意:已知每个人的独一无二的身高以及排在他前面或者后面比他高的人数,问身高字典序最小的排法 分析:首先对身高从矮到高排序,那么可以知道每个人有多少人的身高比他高,那么取较小值(k[i], ...
- HDU 4302 Holedox Eating(multiset)
http://acm.hdu.edu.cn/showproblem.php?pid=4302 题意: 在一条直线上,会有多条命令,如果是0,那么就会在x位置处出现一个蛋糕,如果是1,某人就会找到最近的 ...
- 树链剖分+线段树 HDOJ 5029 Relief grain(分配粮食)
题目链接 题意: 分粮食我就当成涂色了.有n个点的一棵树,在a到b的路上都涂上c颜色,颜色可重复叠加,问最后每一个点的最大颜色数量的颜色类型. 思路: 首先这题的输出是每一个点最后的情况,考虑离线做法 ...
随机推荐
- 【Scrapy】Selectors
Constructing selectors For convenience,response objects exposes a selector on .selector attribute,it ...
- Edmonds 开花算法
Edmonds 开花算法 input: 图G,匹配M,未饱和点u idea: 查找从 u 開始的 M-交错路径.对每一个顶点记录父亲节点. 发现花朵.则收缩. 维护 S 和 T.S 表示沿着已经饱和的 ...
- Hibernate中的自己定义类型——UserType、CompositeUserType
一.UserType Hibernate拥有自己定义映射表属性的机制.主要通过实现接口UserType,详细的UserType: import java.sql.PreparedStatement; ...
- HDU1306 String Matching 【暴力】
String Matching Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) ...
- PHP开发者实用的代码
一.查看邮件是否已被阅读 当你在发送邮件时,你或许很想知道该邮件是否被对方已阅读.这里有段非常有趣的代码片段能够显示对方IP地址记录阅读的实际日期和时间. <? error_reporting( ...
- 一个IM开源项目LiteTalk
http://blog.csdn.net/visualwind/article/details/6086631 http://blog.sina.com.cn/s/blog_54b5ea250101n ...
- hdu 4970 Killing Monsters(数组的巧妙运用) 2014多校训练第9场
pid=4970">Killing Monsters ...
- OpenTK的glutBitmapCharacter的替代方法
由于openTK并没有打包集成GLUT,字体显示就成了一个问题. 办法1:采用QuickFont 优点:可以使用系统自带的字体 缺点:代码尚未成熟.只能2D显示,无法随物体旋转.平移 办法2:同时调用 ...
- Java对象的创建过程
//TODO https://www.cnblogs.com/chenyangyao/p/5296807.html
- C#函数3递归
using System;using System.Collections.Generic;using System.Linq;using System.Text; namespace Console ...