题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4302 , 可以用线段树,也可以STL中的map,multiset,优先队列中的任何一个解决(可我只会线段树QAQ)。

  这道题的解决方法还是比较难想的,我是参考了kuangbin的博客才想到的方法,附链接:http://www.cnblogs.com/kuangbin/archive/2012/08/30/2664289.html


题目解法:

  用线段树来维护区间的最大值与最小值,设当前位置为cur,则对于每次询问,就可以求cur左边区间蛋糕位置的最大值与cur右边区间位置的最小值,选择离cur最近的那个位置即可,此时要更新cur,并且删掉该位置的一个蛋糕,并且调整运动的方向。

  这道题还有几个注意点:这道题的区间是从0到n的,所以具体解的时候要自己调整 ; 初始化线段树中的最大值与最小值 ; 每个位置的蛋糕不止一个,但是每次吃蛋糕的时候只会吃掉一个,所以要注意线段树更新的时候添加和删除蛋糕的个数(我是用一个cnt数组来存每个位置有几个蛋糕,这样可以避免出错) ;如果两边都没有蛋糕的话就停在当前位置 ;最后就是不要忘了运动的方向问题。

#include <iostream>
#include <cstdio>
#include <vector>
#include <cmath>
#include <string>
#include <string.h>
#include <algorithm>
using namespace std;
#define LL __int64
#define eps 1e-8
#define INF INT_MAX
#define lson l , m , rt << 1
#define rson m + 1 , r , rt << 1 | 1
const int MOD = ;
const int maxn = + ;
const int N = + ;
int Max[maxn << ] , Min[maxn << ] , cnt[maxn << ];
void PushUp(int rt)
{
Max[rt] = max(Max[rt << ] , Max[rt << | ]);
Min[rt] = min(Min[rt << ] , Min[rt << | ]);
}
void build(int l , int r , int rt)
{
if(l == r) {
cnt[rt] = ;
Max[rt] = ;
Min[rt] = INF;
return;
}
int m = (l + r) >> ;
build(lson);
build(rson);
PushUp(rt);
}
void update(int pos , int x , int l , int r , int rt)
{
if(l == r) {
if(x) { //添加一个
Max[rt] = x;
Min[rt] = x;
cnt[rt]++;
} else {   //删除一个
if(cnt[rt] == ) {
Max[rt] = ;
Min[rt] = INF;
}
cnt[rt]--;
}
return;
}
int m = (l + r) >> ;
if(pos > m)
update(pos , x , rson);
else
update(pos , x , lson);
PushUp(rt);
}
int query_max(int L , int R , int l , int r , int rt)
{ //询问[L , R]区间的最大值
if(L <= l && R >= r) {
return Max[rt];
}
int m = (l + r) >> ;
if(R <= m)
return query_max(L , R , lson);
else if(L > m)
return query_max(L , R , rson);
else
return max(query_max(L , R , lson) , query_max(L , R , rson));
}
int query_min(int L , int R , int l , int r , int rt)
{ //询问[L , R]区间的最小值
if(L <= l && R >= r) {
return Min[rt];
}
int m = (l + r) >> ;
if(R <= m)
return query_min(L , R , lson);
else if(L > m)
return query_min(L , R , rson);
else
return min(query_min(L , R , lson) , query_min(L , R , rson));
}
void right(int &ans , int &dir , int &cur , int MIN , int n)
{ //向右移动到MIN位置
ans += MIN - cur;
dir = ;
cur = MIN;
update(MIN , , , n + , ); //删掉MIN位置一个蛋糕
}
void left(int &ans , int &dir , int &cur , int MAX , int n)
{ //向左移动到MAX位置
ans += cur - MAX;
dir = ;
cur = MAX;
update(MAX , , , n + , ); //删掉MAX位置一个蛋糕
}
int main()
{
int T , i , n , m , p , ch;
int ans , cur , dir; //结果 , 当前位置 , 运动方向
cin >> T;
for(int k = ; k <= T ; k++)
{
ans = ;
cur = dir = ;
scanf("%d %d" , &n , &m);
build( , n + , );
while(m--) {
scanf("%d" , &ch);
if(!ch) {
scanf("%d" , &p);
update(p + , p + , , n + , );
} else {
int MAX = query_max( , cur , , n + , );
int MIN = query_min(cur , n + , , n + , );
if(MAX == && MIN == INF) {
continue;
} else if(MAX != && MIN == INF) {
left(ans , dir , cur , MAX , n);
} else if(MAX == && MIN != INF) {
right(ans , dir , cur , MIN , n);
} else {
if(MIN - cur < cur - MAX) {
right(ans , dir , cur , MIN , n);
} else if(MIN - cur > cur - MAX) {
left(ans , dir , cur , MAX , n);
} else {
if(dir) {
right(ans , dir , cur , MIN , n);
} else {
left(ans , dir , cur , MAX , n);
}
}
}
}
}
printf("Case %d: %d\n" , k , ans);
}
return ;
}

  PS:等学了STL就用STL把这个题给解决了

  

HDU4302 线段树的更多相关文章

  1. hdu4302 set或者线段树

    题意:       一条蛇生活在一个管子里,然后管子上面的某些位置会一次出现食物,每次蛇都会吃最近的食物,吃完之后就原地不动,等待下一次吃食物,如果有两个食物距离蛇一样远并且都是最近的,那么蛇不会掉头 ...

  2. bzoj3932--可持久化线段树

    题目大意: 最近实验室正在为其管理的超级计算机编制一套任务管理系统,而你被安排完成其中的查询部分.超级计算机中的 任务用三元组(Si,Ei,Pi)描述,(Si,Ei,Pi)表示任务从第Si秒开始,在第 ...

  3. codevs 1082 线段树练习 3(区间维护)

    codevs 1082 线段树练习 3  时间限制: 3 s  空间限制: 128000 KB  题目等级 : 大师 Master 题目描述 Description 给你N个数,有两种操作: 1:给区 ...

  4. codevs 1576 最长上升子序列的线段树优化

    题目:codevs 1576 最长严格上升子序列 链接:http://codevs.cn/problem/1576/ 优化的地方是 1到i-1 中最大的 f[j]值,并且A[j]<A[i] .根 ...

  5. codevs 1080 线段树点修改

    先来介绍一下线段树. 线段树是一个把线段,或者说一个区间储存在二叉树中.如图所示的就是一棵线段树,它维护一个区间的和. 蓝色数字的是线段树的节点在数组中的位置,它表示的区间已经在图上标出,它的值就是这 ...

  6. codevs 1082 线段树区间求和

    codevs 1082 线段树练习3 链接:http://codevs.cn/problem/1082/ sumv是维护求和的线段树,addv是标记这歌节点所在区间还需要加上的值. 我的线段树写法在运 ...

  7. PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树

    #44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...

  8. CF719E(线段树+矩阵快速幂)

    题意:给你一个数列a,a[i]表示斐波那契数列的下标为a[i],求区间对应斐波那契数列数字的和,还要求能够维护对区间内所有下标加d的操作 分析:线段树 线段树的每个节点表示(f[i],f[i-1])这 ...

  9. 【BZOJ-3779】重组病毒 LinkCutTree + 线段树 + DFS序

    3779: 重组病毒 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 224  Solved: 95[Submit][Status][Discuss] ...

随机推荐

  1. jQuery 选择表格(table)里的行和列及改变简单样式

    本文只是介绍如何用jQuery语句对表格中行和列进行选择以及一些简单样式改变,希望它可以对jQuery表格处理的深层学习提供一些帮助jQuery对表格(table)的处理提供了相当强大的功能,比如说对 ...

  2. 洛谷P3803 【模板】多项式乘法(FFT)

    P3803 [模板]多项式乘法(FFT) 题目背景 这是一道FFT模板题 题目描述 给定一个n次多项式F(x),和一个m次多项式G(x). 请求出F(x)和G(x)的卷积. 输入输出格式 输入格式: ...

  3. 2017-10-13 NOIP模拟赛

    入阵曲 #include<iostream> #include<cstdio> #define maxn 401 #ifdef WIN32 #define PLL " ...

  4. 洛谷P3258 [JLOI2014]松鼠的新家

    P3258 [JLOI2014]松鼠的新家 题目描述 松鼠的新家是一棵树,前几天刚刚装修了新家,新家有n个房间,并且有n-1根树枝连接,每个房间都可以相互到达,且俩个房间之间的路线都是唯一的.天哪,他 ...

  5. VBA学习笔记

    这是一个学习VBA编程的学习笔记. 一. 介绍 二. 使用手册 2.1. 如何在Excel2010中开始使用VBA? 2.2. 如何使用VBA编辑器进行编程? 三. 语法说明 3.1 数据类型 3.2 ...

  6. MCP|ZCM|Investigating Lactococcus lactis MG1363 response to phage p2 infection at the proteome level(研究乳酸乳球菌MG1363在噬菌体p2感染后的蛋白质组水平变化)

    一.概述: 噬菌体是特异性感染并最终杀死其细菌宿主的病毒.他们在所有生态系统中发挥着关键的生态作用.尽管经过了几十年的研究,噬菌体与细菌宿主之间的相互作用仍然知之甚少.本研究使用无标记定量蛋白质组学来 ...

  7. c语言中malloc函数的使用

    传送门:https://www.cnblogs.com/shiweihappy/p/4246372.html c语言中内存的管理:https://www.cnblogs.com/tuhooo/p/72 ...

  8. JMETER从JSON响应中提取数据

    如果你在这里,可能是因为你需要使用JMeter从Json响应中提取变量. 好消息!您正在掌握掌握JMeter Json Extractor的权威指南.作为Rest API测试指南的补充,您将学习掌握J ...

  9. 关于苹果出现NaN的情况----由Date格式解析方式不同引起的Bug

    源于一个工作中遇到的问题:IOS 上时间显示为 NaN,而安卓上时间显示正常. 问题的根源在于 安卓 和 苹果 对于 JS Date 对象的不同解析. 安卓: new Date("2018- ...

  10. ACM-较大的数乘法取模技巧*

    比如模数是1e15这种,相乘的时候爆LL了,但是又不想用大数,咋办呢? long long ksc(long long a, long long b, long long mod){ ; while( ...