题意摘自:http://blog.csdn.net/kdqzzxxcc/article/details/9474169 ORZZ

题意:给你N个花瓶,编号是0 到 N - 1 ,初始状态花瓶是空的,每个花瓶最多插一朵花。

然后有2个操作。
操作1,a b c ,往在a位置后面(包括a)插b朵花,输出插入的首位置和末位置。

操作2,a b ,输出区间[a , b ]范围内的花的数量,然后全部清空。

很显然这是一道线段树。区间更新,区间求和,这些基本的操作线段树都可以logN的时间范围内完成。

操作2,很显然就是线段树的区间求和,求出[a , b]范围内的花朵的数量,区间更新,将整个区间全部变成0。

操作1,这里我们首先需要找出他的首位置和末位置,所以需要二分他的位置。

首先我们二分他的首位置, l = a , r = n ,在这个区间内二分,找出第一个0的位置,那就是该操作的首位置pos1。

然后再二分他的末位置,l = pos1 , r = n ,找到第c个0,就是该操作的末位置pos2,然后区间更新[pos1 ,pos2]全部置为1。

tmp = n - minn + 1 - query(minn,n,1) ; 表示【minn,n】区间内空花瓶的个数,如果需要插的花束多余tmp,则只能插tmp束花。

这题重点在于二分找位置,找了半天终于在别人的帮助下调试出来了............

#include <iostream>
#include <algorithm>
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <vector>
#include <set>
#include <queue>
#include <stack>
#include <climits>//形如INT_MAX一类的
#define MAX 100050
#define INF 0x7FFFFFFF
#define REP(i,s,t) for(int i=(s);i<=(t);++i)
#define ll long long
#define mem(a,b) memset(a,b,sizeof(a))
#define mp(a,b) make_pair(a,b)
#define L(x) x<<1
#define R(x) x<<1|1
# define eps 1e-5
//#pragma comment(linker, "/STACK:36777216") ///传说中的外挂
using namespace std; struct node {
int r,l,mid;
int lazy,sum;
} edge[4*MAX]; void up(int num) {
edge[num].sum = edge[L(num)].sum + edge[R(num)].sum;
} void build(int l,int r,int num) {
edge[num].l = l;
edge[num].r = r;
edge[num].mid = (l+r) >> 1;
edge[num].lazy = -1;
if(l == r) {
edge[num].sum = 0;
return ;
}
build(l,edge[num].mid,L(num));
build(edge[num].mid+1,r,R(num));
//
up(num);
} void down(int num) {
//if(edge[num].l == edge[num].r) return ;
if(edge[num].lazy != -1) {
edge[L(num)].lazy = edge[num].lazy;
edge[R(num)].lazy = edge[num].lazy;
edge[L(num)].sum = (edge[L(num)].r - edge[L(num)].l + 1) * edge[num].lazy;
edge[R(num)].sum = (edge[R(num)].r - edge[R(num)].l + 1) * edge[num].lazy;
edge[num].lazy = -1;
} } void update(int l,int r,int num,int k) {
if(l == edge[num].l && r == edge[num].r) {
edge[num].lazy = k;
edge[num].sum = (edge[num].r - edge[num].l + 1) * k;
return ;
}
down(num);
if(r <= edge[num].mid) {
update(l,r,L(num),k);
} else if(l > edge[num].mid) {
update(l,r,R(num),k);
} else {
update(l,edge[num].mid,L(num),k);
update(edge[num].mid + 1,r,R(num),k);
}
up(num);
} int query(int l,int r,int num) {
if(l == edge[num].l && r == edge[num].r) {
return edge[num].sum;
}
down(num);
if(r <= edge[num].mid) {
return query(l,r,L(num));
} else if(l > edge[num].mid) {
return query(l,r,R(num));
} else {
return query(l,edge[num].mid,L(num)) + query(edge[num].mid + 1,r,R(num));
}
} void test(int n) {
for(int i=1; i<=3*n; i++) {
printf("l:%d r:%d sum:%d lazy:%d\n",edge[i].l,edge[i].r,edge[i].sum,edge[i].lazy);
}
}
int main() { int n,m,i,t;
cin >> t;
while(t --) {
scanf("%d%d",&n,&m);
int a,b,c;
build(1,n,1);
for(i=0; i<m; i++) {
scanf("%d%d%d",&a,&b,&c);
if(a == 1) {
int low = b + 1,high = n,mid,minn = 2*n;
if(n - low + 1 - query(low , n , 1) == 0) {
printf("Can not put any one.\n");
continue;
}
while(low <= high) {
mid = (low + high) >> 1;
if(mid - (b + 1) + 1- query(b+1,mid,1) >= 1) {
minn = min(minn,mid);
high = mid - 1;
} else {
low = mid + 1;
}
}
int tmp = n - minn + 1 - query(minn,n,1) ;
if(c >= tmp) c = tmp; ·
low = minn,high = n;
int maxx = INF;
while(low <= high) {
mid = (low + high) >> 1;
tmp = mid - minn + 1 - query(minn,mid,1) ;
if(tmp == c) {
maxx = min(maxx,mid);
high = mid - 1;
} else if(tmp > c) {
high = mid - 1;
} else {
low = mid + 1;
}
}
printf("%d %d\n",minn-1,maxx-1);
update(minn,maxx,1,1); } else {
printf("%d\n",query(b+1,c+1,1));
update(b+1,c+1,1,0);
}
}
puts("");
}
return 0;
}
这题重点在于二分找位置,找了半天终于在别人的帮助下调试出来了............

HDU 4614 Vases and Flowers (2013多校第二场线段树)的更多相关文章

  1. HDU 4614 Vases and Flowers (2013多校2 1004 线段树)

    Vases and Flowers Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others ...

  2. HDU6602 Longest Subarray hdu多校第二场 线段树

    HDU6602 Longest Subarray 线段树 传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6602 题意: 给你一段区间,让你求最长的区间使 ...

  3. HDU 4620 Fruit Ninja Extreme(2013多校第二场 剪枝搜索)

    这题官方结题报告一直在强调不难,只要注意剪枝就行. 这题剪枝就是生命....没有最优化剪枝就跪了:如果当前连续切割数加上剩余的所有切割数没有现存的最优解多的话,不需要继续搜索了 #include &l ...

  4. HDU 4614 Vases and Flowers(线段树+二分)

    Vases and Flowers Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others ...

  5. hdu 4614 Vases and Flowers

    http://acm.hdu.edu.cn/showproblem.php?pid=4614 直接线段树维护 代码: #include<iostream> #include<cstd ...

  6. HDU 4614 Vases and Flowers(二分+线段树区间查询修改)

    描述Alice is so popular that she can receive many flowers everyday. She has N vases numbered from 0 to ...

  7. HDU 4614 Vases and Flowers(线段树+记录区间始末点或乱搞)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4614 题目大意:有n个空花瓶,有两种操作: 操作①:给出两个数字A,B,表示从第A个花瓶开始插花,插B ...

  8. hdu 4614 Vases and Flowers(线段树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4614 题意: 给你N个花瓶,编号是0  到 N - 1 ,初始状态花瓶是空的,每个花瓶最多插一朵花. ...

  9. HDU 4614 Vases and Flowers(线段树+二分)

    题目链接 比赛的时候一直想用树状数组,但是树状数组区间更新之后,功能有局限性.线段树中的lz标记很强大,这个题的题意也挺纠结的. k = 1时,从a开始,插b个花,输出第一个插的位置,最后一个的位置, ...

随机推荐

  1. PHP性能优化学习笔记--PHP周边性能优化--来自慕课网Pangee http://www.imooc.com/learn/205

    PHP一般运行于Linux服务器中,周边主要包括:Linux运行环境.文件存储.数据库.缓存.网络 常见PHP场景的开销次序: 读写内存<<读写数据库(使用内存作为缓存.异步处理)< ...

  2. 45 个非常有用的 Oracle 查询语句(转)

    这里我们介绍的是 40+ 个非常有用的 Oracle 查询语句,主要涵盖了日期操作,获取服务器信息,获取执行状态,计算数据库大小等等方面的查询.这些是所有 Oracle 开发者都必备的技能,所以快快收 ...

  3. MFC获取当前时间

    获取按钮消息响应函数: void CTest17GetTimeDlg::OnGetTime() { // TODO: 在此添加控件通知处理程序代码 //UpdateData(true); CTime  ...

  4. SQLLoader5(从多个数据文件导入到同一张表)

    从多个数据文件导入到同一张表很简单,只需要在INFILE参数指定多个数据文件的路径即可.数据文件1:test1.txt1111 ALLE SALESMAN2222 WARD SALESMAN数据文件2 ...

  5. [HeadFrist-HTMLCSS学习笔记][认识HTML中的“HT”]

    学习超链接 超链接 使用\元素创建一个超文本链接,链接到另一个Web 页面. \元素的内容会变成为Web页面中可单击的文本.href属性告诉浏览器链接的目标文件 <a href="el ...

  6. CentOs7下systemd管理知识要点

    centOs7的一个巨大的变动就是用systemd取代了原来的System V init.systemd是一个完整的软件包,安装完成后有很多物理文件组成,大致分布为,配置文件位于/etc/system ...

  7. 整合 新浪 腾讯 人人 qq空间 分享地址

    function snsShare(snsId, title, content, image, url) { var snsUrl; // 新浪 腾讯 要申请appkey switch (snsId) ...

  8. shell中的退出状态码

    shell中的退出状态码最大只有255,如果超过这个值,就会进行取余运算,即如果执行如下命令: exit exitCode 如果exitCode大于255,那么实际的状态码为exitCode % 25 ...

  9. Mysql常用命令行大全(转)

    第一招.mysql服务的启动和停止 net stop mysql net start mysql 第二招.登陆mysql 语法如下: mysql -u用户名 -p用户密码 键入命令mysql -uro ...

  10. JQuery validator扩展

    //validator 扩展 jQuery.validator.addMethod("mail", function(value, element, messages) { ret ...