K大数查询

【问题描述】

有N个位置,M个操作。操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是多少。

【输入格式】

第一行N,M

接下来M行,每行形如1 a b c或2 a b c

【输出格式】

输出每个询问的结果

【样例输入】

2 5
1 1 2 1
1 1 2 2
2 1 1 2
2 1 1 1
2 1 2 3

【样例输出】

1
2
1

【样例说明】

第一个操作 后位置 1 的数只有 1 , 位置 2 的数也只有 1 。

第二个操作 后位置 1的数有 1 、 2 ,位置 2 的数也有 1 、 2 。

第三次询问 位置 1 到位置 1 第 2 大的数 是1 。

第四次询问 位置 1 到位置 1 第 1 大的数是 2 。

第五次询问 位置 1 到位置 2 第 3大的数是 1 。‍

【数据范围】

N,M<=50000,N,M<=50000,a<=b<=N

1操作中abs(c)<=N,2操作中c<=Maxlongint


题解:

我们将询问离线,做整体二分

题目中有负数,那么我们转化一下,将每个数变为n-i+1,输出答案时再变为n-ans+1

对于一个操作1,如果这个操作加入的c是不超过mid的

用线段树在区间内加1,表示此区间小于等于mid的数多了一个,那么将它放置到左区间

否则将其放置到右区间,表示这个操作的贡献在右区间

对于一个操作2,查询在区间内小于等于mid的数的个数tot

如果tot超过k,将其放置到左区间,表示答案在左区间

否则将k减去tot,放置到右区间,表示需要在右区间找k-tot大的数

 #include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
using namespace std;
struct S { long long x, y, z, id, flag; } a[], c[];
bool lr[];
long long n, m, tot, maxx = -;
long long sum[], ans[], node[];
void Down(long long k, int l, int r)
{
if(node[k] != )
{
int mi = (l + r) >> ;
node[k * ] += node[k];
node[k * + ] += node[k];
sum[k * ] += node[k] * (mi - l + );
sum[k * + ] += node[k] * (r - mi);
node[k] = ;
}
}
void Inc(int k, int l, int r, int x, int y, int z)
{
if(x <= l && r <= y)
{
sum[k] += z * (r - l + );
node[k] += z;
return;
}
Down(k, l, r);
int mi = (l + r) >> ;
if(mi >= x) Inc(k * , l, mi, x, y, z);
if(mi < y) Inc(k * + , mi + , r, x, y, z);
sum[k] = sum[k * ] + sum[k * + ];
}
long long Sum(long long k, long long l, long long r, long long x, long long y)
{
if(x <= l && r <= y) return sum[k];
Down(k, l, r);
long long mi = (l + r) >> , res = ;
if(mi >= x) res += Sum(k * , l, mi, x, y);
if(mi < y) res += Sum(k * + , mi + , r, x, y);
return res;
}
void Two(long long x, long long y, long long l, long long r)
{
cout<<x<<' '<<y<<' '<<l<<' '<<r<<endl;
long long mi = (l + r) >> ;
if(l == r)
{
for(int i = x; i <= y; ++i)
if(a[i].flag)
ans[a[i].id] = mi;
return;
}
long long temp, s = x;
for(int i = x; i <= y; ++i)
{
if(a[i].flag)
{
temp = Sum(, , * n + , a[i].x, a[i].y);
if(temp < a[i].z)
{
a[i].z -= temp;
lr[i] = false;
}
else
{
++s;
lr[i] = true;
}
}
else
{
if(a[i].z <= mi)
{
Inc(, , * n + , a[i].x, a[i].y, );
++s;
lr[i] = true;
}
else lr[i] = false;
}
}
for(int i = x; i <= y; ++i)
if(!a[i].flag && a[i].z <= mi)
Inc(, , * n + , a[i].x, a[i].y, -);
long long o = x;
for(int i = x; i <= y; ++i)
if(lr[i]) c[o++] = a[i];
else c[s++] = a[i];
for(int i = x; i <= y; ++i) a[i] = c[i];
Two(x, o - , l, mi), Two(o, y, mi + , r);
}
int main()
{
scanf("%lld%lld", &n, &m);
for(int i = ; i <= m; ++i)
{
scanf("%lld%lld%lld%lld", &a[i].flag, &a[i].x, &a[i].y, &a[i].z);
--a[i].flag;
if(!a[i].flag)
{
a[i].z = n - a[i].z + ;
maxx = (a[i].z > maxx) ? a[i].z : maxx;
}
else a[i].id = ++tot;
}
Two(, m, , maxx);
for(int i = ; i <= tot; ++i) printf("%lld\n", n - ans[i] + );
}

K大数查询 BZOJ 3110的更多相关文章

  1. 【ZJOI2013】k大数查询 BZOJ 3110

    Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个位置到第b个位 ...

  2. BZOJ 3110 K大数查询 | 整体二分

    BZOJ 3110 K大数查询 题面 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个 ...

  3. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

  4. BZOJ 3110: [Zjoi2013]K大数查询 [树套树]

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 6050  Solved: 2007[Submit][Sta ...

  5. 树套树专题——bzoj 3110: [Zjoi2013] K大数查询 &amp; 3236 [Ahoi2013] 作业 题解

    [原题1] 3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MB Submit: 978  Solved: 476 Descri ...

  6. bzoj 3110: [Zjoi2013]K大数查询 树状数组套线段树

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1384  Solved: 629[Submit][Stat ...

  7. BZOJ 3110: [Zjoi2013]K大数查询( 树状数组套主席树 )

    BIT+(可持久化)权值线段树, 用到了BIT的差分技巧. 时间复杂度O(Nlog^2(N)) ---------------------------------------------------- ...

  8. BZOJ 3110([Zjoi2013]K大数查询-区间第k大[段修改,在线]-树状数组套函数式线段树)

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec   Memory Limit: 512 MB Submit: 418   Solved: 235 [ Submit][ ...

  9. BZOJ 3110 [Zjoi2013]K大数查询(整体二分)

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 11654  Solved: 3505[Submit][St ...

随机推荐

  1. django连接Oracle过程中出现的问题

    开始时版本信息: python 3.6   +   ce_oracle 6 最终版本信息: python 3.5   + ce_oracle 5.2 ce_oracle版本问题 cx_Oracle-5 ...

  2. Android之通过adb shell 模拟器 error: more than one device and emulator 改ip dns

    error: more than one device and emulator 如果出现上面那种情况 请关闭  ide 输入下面的  再次重新启动 模拟器 如果实际上只有一个设备或模拟器,并且查到有 ...

  3. visibilitychange 标签可见性

    var pageVisibility = document.visibilityState;// 监听 visibility change 事件document.addEventListener('v ...

  4. 有趣的this以及apply,call,bind方法

    看this指向谁,要看执行时而非定义时(箭头函数除外).函数没有绑定在对象上调用,非'strict'模式下,this指向window,否则为undefined 改变this指向的方法 1. apply ...

  5. 博弈论入门 Bash 、Nim 、Wythoff's Game结论及c++代码实现

    SG函数先不说,给自己总结下三大博弈.和二进制及黄金分割联系密切,数学真奇妙,如果不用考试就更好了. 1.Bash Game:n个物品,最少取1个,最多取m个,先取完者胜. 给对手留下(m+1)的倍数 ...

  6. 概述「并查集补集转化」模型&&luoguP1330 封锁阳光大学

    奇妙的模型转化以及并查集思想 模型概述 有图$G=(V,E)$,初始所有点为白色,现在要将其中一些点染为黑色,要求染色后满足:$∀(u,v)∈E$,$∃col_u!=col_v$.求最小染色点数. 题 ...

  7. [LUOGU] P1387 最大正方形

    题目描述 在一个n*m的只包含0和1的矩阵里找出一个不包含0的最大正方形,输出边长. 输入输出格式 输入格式: 输入文件第一行为两个整数n,m(1<=n,m<=100),接下来n行,每行m ...

  8. 【OS_Linux】Linux系统中目录及文件管理

    1.Linux系统中目录的树状结构 目录 /bin 存放二进制可执行文件(ls,cat,mkdir等),常用命令一般都在这里. /etc 存放系统管理和配置文件 /home 存放所有用户文件的根目录, ...

  9. django(django项目创建,数据库迁移)

    Django项目的创建与介绍 安装:pip3 install django==1.11 查看版本号:django-admin --version 新建项目: 1.切到目标目录 2.django-adm ...

  10. 剑指Offer(书):调整数组顺序使奇数位于偶数前面

    题目:输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有的奇数位于数组的前半部分,所有的偶数位于数组的后半部分,并保证奇数和奇数,偶数和偶数之间的相对位置不变. public void ...