K大数查询 BZOJ 3110
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的更多相关文章
- 【ZJOI2013】k大数查询 BZOJ 3110
Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个位置到第b个位 ...
- BZOJ 3110 K大数查询 | 整体二分
BZOJ 3110 K大数查询 题面 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c 如果是2 a b c形式,表示询问从第a个 ...
- [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)
[BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...
- BZOJ 3110: [Zjoi2013]K大数查询 [树套树]
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6050 Solved: 2007[Submit][Sta ...
- 树套树专题——bzoj 3110: [Zjoi2013] K大数查询 & 3236 [Ahoi2013] 作业 题解
[原题1] 3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MB Submit: 978 Solved: 476 Descri ...
- bzoj 3110: [Zjoi2013]K大数查询 树状数组套线段树
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1384 Solved: 629[Submit][Stat ...
- BZOJ 3110: [Zjoi2013]K大数查询( 树状数组套主席树 )
BIT+(可持久化)权值线段树, 用到了BIT的差分技巧. 时间复杂度O(Nlog^2(N)) ---------------------------------------------------- ...
- BZOJ 3110([Zjoi2013]K大数查询-区间第k大[段修改,在线]-树状数组套函数式线段树)
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MB Submit: 418 Solved: 235 [ Submit][ ...
- BZOJ 3110 [Zjoi2013]K大数查询(整体二分)
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 11654 Solved: 3505[Submit][St ...
随机推荐
- nodejs:遍历文件夹文件统计文件大小
根据 http://blog.csdn.net/hero82748274/article/details/45700465这里的思路对读写文件做了一个 封装: webpack在打包的时候可以借助ass ...
- Hopfield 网络(上)
讲的什么 这部分主要对 Hopfield 网络作一大概的介绍.写了其模型结构.能量函数和网络的动作方式.主要参考了网上搜到的一些相关 PPT. 概述 早在 1982 年,Hopfield 发表的文 ...
- c语言文件打开模式
(转载) 在C语言的文件操作语法中,打开文件文件有以下12种模式,如下图: 打开模式 只可以读 只可以写 读写兼备 文本模式 r w a r+ w+ a+ 二进制模式 rb wb ab rb ...
- js截屏
<html><head> <meta name="layout" content="main"> <meta http ...
- C++ STL容器之 map
map 是一种有序无重复的关联容器. 关联容器与顺序容器不同,他们的元素是按照关键字来保存和访问的,而顺序元素是按照它们在容器中的位置保存和访问的. map保存的是一种 key - value 的pa ...
- 利用Resttemplate进行put请求
开发中,最常用的是post.get这两种.今天我给大家展示一个利用put请求的demo,其实put请求跟post请求没啥区别,但是没有返回值. void put(String var1, @Nulla ...
- tiny4412 busybox制作根文件系统rootfs nfs 挂载 ubuntu 14.04
http://blog.csdn.net/liudijiang/article/details/50555429(转) 首先得要有制作好的uboot和linux内核镜像zImage,先烧录到sd卡里, ...
- Lex与Yacc学习(六)之lex & yacc (简单计算器程序) 运行
词法分析程序ch3-01.l %{ #include "ch3-01.tab.h" extern int yylval; %} %% [0-9]+ { yylval = atoi( ...
- 牛客网暑期ACM多校训练营(第六场) I Team Rocket(线段树)
题意: 给定n个区间, m次询问, 每次询问给一个点, 问这个点在哪些区间内, 然后删掉这些区间. 分析: 将n个区间按L大小升序排列, 然后将这些区间视为点构建一棵n个点的线段树, 树的节点记录这个 ...
- mac securecrt自动保存密码
一.问题描述 mac有自带的终端,可以运行ssl和sftp,但是目录操作,文件操作和文件上传是分开的,很不方便,并且文件上传命令需要文件的全路路径. 使用securecrt能方便的解决上述的问题,并且 ...