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颜色,颜色可重复叠加,问最后每一个点的最大颜色数量的颜色类型. 思路: 首先这题的输出是每一个点最后的情况,考虑离线做法 ...
随机推荐
- 【转】Web Worker javascript多线程编程(一)
原文:https://www.cnblogs.com/peakleo/p/6218823.html -------------------------------------------------- ...
- Random Forest 与 GBDT 的异同
曾经在看用RF和GBDT的时候,以为是非常相似的两个算法,都是属于集成算法,可是细致研究之后,发现他们根本全然不同. 以下总结基本的一些不同点 Random Forest: bagging (你懂得. ...
- 剑指Offer面试题15(Java版):链表中倒数第K个结点
题目: 输入一个链表.输出该链表中倒数第k哥结点. 为了符合大多数人的习惯,本题从1開始计数.即链表的尾结点是倒数第1个结点. 比如一个链表有6个结点.从头结点開始它们的值依次是1.2.3,4,5, ...
- Visual Studio 2017中使用正则修改部分内容 如何使用ILAsm与ILDasm修改.Net exe(dll)文件 C#学习-图解教程(1):格式化数字字符串 小程序开发之图片转Base64(C#、.Net) jquery遍历table为每一个单元格取值及赋值 。net加密解密相关方法 .net关于坐标之间一些简单操作
Visual Studio 2017中使用正则修改部分内容 最近在项目中想实现一个小工具,需要根据类的属性<summary>的内容加上相应的[Description]特性,需要实现的效 ...
- Android5.0(lollipop)新特性介绍(一)
今年6月的Google I/O大会上.Android L的初次见面我相信让会让非常多android粉丝有些小激动和小期待.当然作为开发人员的我来说,激动不言而喻,毕竟这是自08年以来改变最大的一个版本 ...
- 一个很小的C++写的MVC的例子
#include<iostream> #include<vector> //get namespace related stuff using std::cin; using ...
- 4 Ionic导航和核心组件--旅游应用
简介:在本节课中,我们将会通过一个虚构的旅游景点来构建一款功能完善的应用.本应用的核心特性是管理用户的应用内导航.本节课的主要目的,是展现构建一个完整的应用的过程. 无论是什么移动应用,最重要的功能之 ...
- UITableViewController的子控件不随着滑动
UITableViewController的子控件不随着滑动 我们知道有时候使用UITableViewController简单便捷,省事,但是如果我们使用了addSubview,无论是[self.vi ...
- hdu3555(数位DP dfs/递推)
Bomb Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 131072/65536 K (Java/Others)Total Submi ...
- asp.net MVC5 中的捆绑和更改bootstap默认的样式
在MVC5的视图中使用@Scritps.Render(),@Styles.Render() 分别可以加载样式和脚本.捆绑的和实际的路径都可以. 并且可以利用 编程的方式灵活引用css文件和脚本文件. ...