链接:http://acm.hdu.edu.cn/showproblem.php?pid=2871

题意:

四种操作:

1.Reset  清空所有内存
2.New x  分配一个大小为x的内存块返回,返回能分配的最小的起始点

3.Free x  释放当前点所在的内存块,并输出左右端点

4.Get x  返回第x个内存块的起始点

讨论每个操作的写法:

第一个操作,把线段树初始化就好了

第二个操作,区间合并的基础操作,

第三个操作:多维护两个数组:st,ed代表当前点所属内存块的左右区间

第四个操作:再开棵树,存每个内存块的起始点,每个起始点下标+1,然后直接找第k大就好了。注意每次2,3,操作内存块发生变化时记得更新这个树上的信息。

题面说了每组数据中间要有空行,不然会PE.

实现代码:

#include<bits/stdc++.h>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define mid int m = (l + r) >> 1
#define ls rt<<1
#define rs rt<<1|1
const int M = 1e5 + ;
int lsum[M<<],rsum[M<<],sum[M<<],lazy[M<<],st[M<<],ed[M<<];
int cnt[M<<],cov[M<<],n;
void pushup(int l,int r,int rt){
mid;
lsum[rt] = lsum[ls]; rsum[rt] = rsum[rs];
sum[rt] = max(sum[ls],sum[rs]);
if(lsum[rt] == m-l+) lsum[rt] += lsum[rs];
if(rsum[rt] == r-m) rsum[rt] += rsum[ls];
sum[rt] = max(rsum[ls]+lsum[rs],sum[rt]);
} void pushdown(int l,int r,int rt){
if(lazy[rt]!=-){
mid;
lazy[ls] = lazy[rs] = lazy[rt];
sum[ls] = lsum[ls] = rsum[ls] = (m-l+)*lazy[rt];
sum[rs] = lsum[rs] = rsum[rs] = (r-m)*lazy[rt];
st[ls] = st[rs] = st[rt];
ed[ls] = ed[rs] = ed[rt];
lazy[rt] = -;
}
} void update(int L,int R,int c,int l,int r,int rt){
if(L <= l&&R >= r){
lazy[rt] = c;
lsum[rt] = rsum[rt] = sum[rt] = (r-l+)*c;
if(c) st[rt] = ed[rt] = -;
else{
st[rt] = L; ed[rt] = R;
}
return ;
}
pushdown(l,r,rt);
mid;
if(L <= m) update(L,R,c,lson);
if(R > m) update(L,R,c,rson);
pushup(l,r,rt);
} void rebuild(){
update(,n,,,n,);
//cout<<sum[1]<<endl;
cnt[] = ;
cov[] = ;
} int New(int p,int l,int r,int rt){ //返回空位的左边界
if(l == r) return l;
mid;
pushdown(l,r,rt);
if(sum[ls] >= p) return New(p,lson);
else if(rsum[ls]+lsum[rs]>=p) return m-rsum[ls]+;
else return New(p,rson);
} int Free(int p,int l,int r,int rt){
if(l == r) return rt;
mid;
pushdown(l,r,rt);
if(p <= m) return Free(p,lson);
else return Free(p,rson);
} void up(int rt){
cnt[rt] = cnt[ls] + cnt[rs];
} void down(int rt){
if(cov[rt]){
cnt[ls] = cnt[rs] = ;
cov[ls] = cov[rs] = ;
cov[rt] = ;
}
} int get(int p,int l,int r,int rt){
if(l == r) return l;
mid;
down(rt);
if(cnt[ls] >= p) return get(p,lson);
else return get(p-cnt[ls],rson);
} void change(int p,int c,int l,int r,int rt){
if(l == r){
cnt[rt] = c;
return;
}
down(rt);
mid;
if(p <= m) change(p,c,lson);
else change(p,c,rson);
up(rt);
} int main()
{
int q,k,x;
while(scanf("%d%d",&n,&q)!=EOF){
rebuild();
while(q--){
char op[];
scanf("%s",op);
if(op[] == 'N'){
scanf("%d",&x);
//cout<<sum[1]<<" "<<x<<endl;
if(sum[] < x) printf("Reject New\n");
else{
int k = New(x,,n,);
printf("New at %d\n",k);
change(k,,,n,);
update(k,k+x-,,,n,);
}
}
else if(op[]=='F'){
scanf("%d",&x);
int k = Free(x,,n,);
if(st[k] < ) printf("Reject Free\n");
else{
printf("Free from %d to %d\n",st[k],ed[k]);
change(st[k],,,n,);
update(st[k],ed[k],,,n,);
}
}
else if(op[] == 'R'){
rebuild();
printf("Reset Now\n");
}
else{
scanf("%d",&x);
if(x > cnt[])
printf("Reject Get\n");
else
printf("Get at %d\n",get(x,,n,));
}
}
printf("\n");
}
return ;
}

hdu 2871 Memory Control (区间合并 连续段的起始位置 点所属段的左右端点)的更多相关文章

  1. hdu 2871 Memory Control(伸展树splay tree)

    hdu 2871 Memory Control 题意:就是对一个区间的四种操作,NEW x,占据最左边的连续的x个单元,Free x 把x单元所占的连续区间清空 , Get x 把第x次占据的区间输出 ...

  2. hdu 2871 Memory Control(线段树)

    题目链接:hdu 2871 Memory Control 题目大意:模拟一个内存分配机制. Reset:重置,释放全部空间 New x:申请内存为x的空间,输出左地址 Free x:释放地址x所在的内 ...

  3. HDU 2871 Memory Control

    一共4种操作 其中用线段树 区间合并,来维护连续空的长度,和找出那个位置.其他用vector维护即可 #include<cstring> #include<cstdio> #i ...

  4. ●HDU 2871 Memory Control(Splay)

    ●赘述题目 四种操作: ○Reset:将整个内存序列清空. ○New a:在尽量靠左的位置新建一个长度为a的内存块,并输出改内存块起始位置.(各个内存块即使相邻也不会合并..) ○Free a:将a点 ...

  5. HDU 2871"Memory Control"(线段树区间和并+set.lower_bound)

    传送门 •题意 有 n 个内存单元(编号从1开始): 给出 4 种操作: (1)Reset :表示把所有的内存清空,然后输出 "Reset Now". (2)New x :表示申请 ...

  6. HDU 3911 线段树区间合并、异或取反操作

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...

  7. HDU 4553 约会安排 (区间合并)【线段树】

    <题目链接> 寒假来了,又到了小明和女神们约会的季节.  小明虽为屌丝级码农,但非常活跃,女神们常常在小明网上的大段发言后热情回复“呵呵”,所以,小明的最爱就是和女神们约会.与此同时,也有 ...

  8. hdu 3308(线段树区间合并)

    LCIS Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  9. HDU 3308 LCIS (经典区间合并)【线段树】

    <题目链接> 题目大意: 给你一段序列,对其进行两种操作,一是修改某个序号的点的值:二是查询某个区间的LCIS(最长上升子序列). 解题分析: 线段树区间合并的典型例题,用求某个区间的LC ...

随机推荐

  1. 51nod 1594 Gcd and Phi 反演

    OTZ 又被吊打了...我当初学的都去哪了??? 思路:反演套路? 提交:\(1\)次 题解: 求\(\sum_{i=1}^{n}\sum_{j=1}^{n}\varphi(gcd(\varphi(i ...

  2. 在使用Telnet连接localhost时所遇到的问题:出现 ‘telnet’ 不是内部或外部命令,也不是可运行的程序或批处理文件

    1.出现 ‘telnet’ 不是内部或外部命令,也不是可运行的程序或批处理文件.原因:因为本机的Telnet客户端默认是关闭的,所以我们要手动打开解决方案:打开控制面板–>程序–>打开或关 ...

  3. ckeditor不能粘贴word的问题如何解决

    自动导入Word图片,或者粘贴Word内容时自动上传所有的图片,并且最终保留Word样式,这应该是Web编辑器里面最基本的一个需求功能了.一般情况下我们将Word内容粘贴到Web编辑器(富文本编辑器) ...

  4. BZOJ3791 作业 动态规划

    你发现染 $k$ 次最多会将这个序列分成 $2k-1$ 段,然后任何 $2k-1$ 段以内的方案一定能被构建出来,所以直接 dp 就好了 #include <bits/stdc++.h> ...

  5. vue中setInterval的清除

    两种清除setInterval的方式: 方案一: data() { return { timer: null // 定时器名称 } }, mouted() { this.timer = (() =&g ...

  6. 【概率论】3-7:多变量分布(Multivariate Distributions Part II)

    title: [概率论]3-7:多变量分布(Multivariate Distributions Part II) categories: Mathematic Probability keyword ...

  7. 经过测试,feign只能通过@RequestBody传对象参数

    通过feign调用,使用ModelAttribute 注解,参数没法传到对应的server

  8. 字典-Python基础前传(9)

    (一)Python中为什么要有字典 jacky说科学存在的逻辑只有两个: 1.解释问题 2.解决问题 我们明白了科学的逻辑,我们理解任何的知识和技能,都是很简单的 之前jacky跟大家说list因为太 ...

  9. MySQL数据分析(7)-试着使用SQL

    (一) 1.1 启动服务器 Windows版命令: net start mysql 或者 C:\mysql-5.5.20-winx64\mysql-5.5.20-winx64\mysql Mac版命令 ...

  10. gitlab配置邮箱postfix(新用户激活邮件)

    亲测可用 https://www.cnblogs.com/yoyoketang/p/10287345.html