/*
hdu4614
本题刚开始想能不能记录该区间最前面开始的点,最后面的点,区间空的数量;但是病不行
然后线段树的本质是区间操作,所以!这题主要就是区间的空的全放满,只要定出区间的边界就好办了;
这里用二分查找的方法,现计算满足数量的区间的尾,因为头已经确定了,及时不放花也是确定的,只要靠数量定出尾就可以了;
然后就是区间修改了;
*/
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define MAX_N 50001
using namespace std;
int n,m,first,last,num,x;
struct tree
{
int l,r,fir,las,s,same,val;
}tr[MAX_N*];
void build(int rt,int l,int r)
{
tr[rt].l=l;tr[rt].r=r;
tr[rt].fir=l;
tr[rt].las=r;
tr[rt].s=;
tr[rt].val=r-l+;
if(l==r)
{
tr[rt].same=;
return ;
}
tr[rt].same=;
int mid=(l+r)/;
build(rt<<,l,mid);
build(rt<<|,mid+,r);
}
void Pushup(int rt)
{
int l=rt<<,r=rt<<|;
if(tr[l].fir==)
tr[rt].fir=tr[r].fir;
else tr[rt].fir=tr[l].fir;
if(tr[r].las==)
tr[rt].las=tr[l].las;
else tr[rt].las=tr[r].las;
tr[rt].val=tr[r].val+tr[l].val; }
void Pushdown(int rt)
{
if(tr[rt].l==tr[rt].r)
return ;
int l=rt<<,r=rt<<|;
if(tr[rt].same)
{
if(tr[rt].s==)
{
tr[r].val=tr[l].val=;
tr[r].s=tr[l].s=;
tr[r].fir=tr[l].fir=;
tr[r].las=tr[l].las=;
tr[r].same=tr[l].same=;
tr[rt].same=;
}
else
{
tr[r].val=tr[r].r-tr[r].l+;
tr[l].val=tr[l].r-tr[l].l+;
tr[r].s=tr[l].s=;
tr[r].same=tr[l].same=;
tr[r].fir=tr[r].l;
tr[r].las=tr[r].r;
tr[l].fir=tr[l].l;
tr[l].las=tr[l].r;
tr[rt].same=;
}
}
}
void Update1(int rt,int l,int r)
{
if(x<=)
return ;
if(tr[rt].val==)
return ;//在二分查找定区间后,少了这一句所以一直tle
if(tr[rt].l==l&&tr[rt].val>)
{
if(first==)
first=tr[rt].fir;
if(tr[rt].val<=x)
{
if(last<tr[rt].las)
last=tr[rt].las;
x-=tr[rt].val;
tr[rt].same=;
tr[rt].s=;
tr[rt].val=;
tr[rt].fir=;
tr[rt].las=;
return ;
}
}
if(tr[rt].l==tr[rt].r)
return ;
Pushdown(rt);
int L=rt<<,R=rt<<|;
if(l<=tr[L].r)
{
if(r<=tr[L].r)
Update1(L,l,r);
else
Update1(L,l,tr[L].r);
}
if(r>=tr[R].l)
{
if(l>=tr[R].l)
Update1(R,l,r);
else
Update1(R,tr[R].l,r);
}
Pushup(rt);
}
void Update2(int rt,int l,int r)
{
if(tr[rt].l==l&&tr[rt].r==r)
{
num+=r-l+-tr[rt].val;
tr[rt].val=r-l+;
tr[rt].s=;
tr[rt].same=;
tr[rt].fir=l;
tr[rt].las=r;
return;
}
if(tr[rt].l==tr[rt].r)
return ;
Pushdown(rt);
int L=rt<<,R=rt<<|;
if(l<=tr[L].r)
{
if(r<=tr[L].r)
Update2(L,l,r);
else
Update2(L,l,tr[L].r);
}
if(r>=tr[R].l)
{
if(l>=tr[R].l)
Update2(R,l,r);
else
Update2(R,tr[R].l,r);
}
Pushup(rt);
}
int sum(int rt,int l,int r)
{
if(tr[rt].l==l&&tr[rt].r==r)
{
return tr[rt].val;
}
int ans=;
Pushdown(rt);//因为只是查询,所以区间整体并没有变,就没有必要pushup了;
int L=rt<<,R=rt<<|;
if(l<=tr[L].r)
{
if(r<=tr[L].r)
ans+= sum(L,l,r);
else
ans+= sum(L,l,tr[L].r);
}
if(r>=tr[R].l)
{
if(l>=tr[R].l)
ans+= sum(R,l,r);
else
ans+= sum(R,tr[R].l,r);
}
return ans;
}
int bisearch(int a,int f)
{
if(sum(,a,n)==)
return -;
if(sum(,a,n)<f)//此处的等于号的问题;
return n;
int l=a,r=n;
int ans=a;
while(l<=r)
{
int mid=(l+r)/;
if(sum(,a,mid)>=f)//以及此处的等于号;
{
ans=mid;//这里ans的取值;
r=mid-;
}
else l=mid+;
}
return ans;
}
int main()
{
int T;
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m);
build(,,n);
for(int i=;i<m;i++)
{
int ty,z,y;
scanf("%d%d%d",&ty,&z,&y);
if(ty==)
{
int t=bisearch(z+,y);
if(t!=-)
{
x=y;
first=;
last=;
Update1(,z+,t);
printf("%d %d\n",first-,last-);
}
else
printf("Can not put any one.\n");
}
else
{
num=;
Update2(,z+,y+);
printf("%d\n",num);
}
}
printf("\n");
}
return ;
}

kb-07线段树-12--二分查找区间边界的更多相关文章

  1. HDU 4614 线段树+二分查找

    Vases and Flowers 题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=4614 Problem Description Alice is s ...

  2. K-th Number 线段树(归并树)+二分查找

    K-th Number 题意:给定一个包含n个不同数的数列a1, a2, ..., an 和m个三元组表示的查询.对于每个查询(i, j, k), 输出ai, ai+1, ... ,aj的升序排列中第 ...

  3. poj 2892---Tunnel Warfare(线段树单点更新、区间合并)

    题目链接 Description During the War of Resistance Against Japan, tunnel warfare was carried out extensiv ...

  4. BZOJ3110 [Zjoi2013]K大数查询 树套树 线段树 整体二分 树状数组

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ3110 题意概括 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位 ...

  5. BZOJ5142: [Usaco2017 Dec]Haybale Feast 线段树或二分答案

    Description Farmer John is preparing a delicious meal for his cows! In his barn, he has NN haybales ...

  6. POJ 3468 A Simple Problem with Integers(线段树 成段增减+区间求和)

    A Simple Problem with Integers [题目链接]A Simple Problem with Integers [题目类型]线段树 成段增减+区间求和 &题解: 线段树 ...

  7. 【BZOJ4552】排序(线段树,二分答案)

    [BZOJ4552]排序(线段树,二分答案) 题面 BZOJ 题解 好神的题啊 直接排序我们做不到 怎么维护? 考虑一下,如果我们随便假设一个答案 怎么检验它是否成立? 把这个数设成\(1\),其他的 ...

  8. HDU 3577 Fast Arrangement ( 线段树 成段更新 区间最值 区间最大覆盖次数 )

    线段树成段更新+区间最值. 注意某人的乘车区间是[a, b-1],因为他在b站就下车了. #include <cstdio> #include <cstring> #inclu ...

  9. luoguP6619 [省选联考 2020 A/B 卷]冰火战士(线段树,二分)

    luoguP6619 [省选联考 2020 A/B 卷]冰火战士(线段树,二分) Luogu 题外话1: LN四个人切D1T2却只有三个人切D1T1 很神必 我是傻逼. 题外话2: 1e6的数据直接i ...

随机推荐

  1. python基础教程总结12——数据库

    1. Python 数据库 API 很多支持SQL标准的数据库在Python中都有对应的客户端模块.为了在提供相同功能(基本相同)的不同模块之间进行切换(兼容),Python 规定了一个标准的 DB ...

  2. 关于JavaScript的变量和函数提升

    第一种理解方式:let和const不能被使用,直到他们被声明 对于var定义的变量,解析器会提升其到作用域顶部. // Outputs: undefined console.log(x); var x ...

  3. 关系代数演算So Easy

    关系代数运算So Easy 关系代数是以关系为运算的一组高级运算的集合.由于定义为属性个数 相同的元组的集合,因此集合代数的操作就可以引入到关系代数中.关系代数也可以看做是一种抽象的查询语言,是对关系 ...

  4. 校招准备-关系型数据库与nosql

    深入理解常见的数据库的设计架构, 其中用到的数据结构, 算法等 SQL执行流程和优化, 可以了解一下calcite: https://calcite.apache.org/

  5. MySql下最好用的数据库管理工具是哪个

    MySql下最好用的数据库管理工具是哪个? 维基上有个很全的列表: https://en.wikipedia.org/wiki/Comparison_of_database_tools   1. ph ...

  6. oracle 快速复制一张表,并在此创建索引,日志及并行度

    复制表结构及其数据 create table table_name_new as select * from table_name_old 只复制表结构 create table table_name ...

  7. Python学习笔记5(函数)

    [摘要]本文详细介绍python中的函数,以及与之相关的参数和作用域的概念,并介绍递归的概念以及在程序中的应用. 函数定义 定义函数要用函数定义语句def.如下: def hello(name): r ...

  8. NOIP模拟赛 czy的后宫4

    czy的后宫4 [问题描述] czy有很多妹子,妹子虽然数量很多,但是质量不容乐观,她们的美丽值全部为负数(喜闻乐见). czy每天都要带N个妹子到机房,她们都有一个独一无二的美丽值,美丽值为-1到- ...

  9. Bootstrap 模态框 禁止点击空白关闭模态框事件

    在模态框的div中加上 aria-hidden="true" data-backdrop="static" <div class="modal ...

  10. DeepFaceLab小白入门(6):脸部替换以及合成视频!

    前面的都是准备工作,这个环节才是真的换脸.换脸主要分两部分,1,图片换脸,2,把图片合成视频. 7) convert H64 debug.bat 这个环节是和训练环节相对于的,比如我们之前选的是H64 ...