BZOJ3110 K大数查询 【线段树 + 整体二分 或 树套树(非正解)】
Description
有N个位置,M个操作。操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c
如果是2 a b c形式,表示询问从第a个位置到第b个位置,第C大的数是多少。
Input
第一行N,M
接下来M行,每行形如1 a b c或2 a b c
Output
输出每个询问的结果
Sample Input
1 1 2 1
1 1 2 2
2 1 1 2
2 1 1 1
2 1 2 3
Sample Output
2
1
HINT
【样例说明】
第一个操作 后位置 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
题解
- #include<iostream>
- #include<cstdio>
- #include<cstring>
- #include<algorithm>
- #define LL long long int
- #define REP(i,n) for (int i = 1; i <= (n); i++)
- #define fo(i,x,y) for (int i = (x); i <= (y); i++)
- #define Redge(u) for (int k = head[u]; k != -1; k = edge[k].next)
- using namespace std;
- const int maxn = 400005,maxm = 20000005,INF = 1000000000;
- inline int read(){
- int out = 0,flag = 1;char c = getchar();
- while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
- while (c >= 48 && c <= 57) {out = out * 10 + c - 48; c = getchar();}
- return out * flag;
- }
- int a,b,c,N;
- int n,m,siz = 0;
- int rt[maxn];
- int ls[maxm],rs[maxm],sum[maxm],lazy[maxm],L,R;
- inline void pd(int u,int l,int r){
- if (!lazy[u] || l == r) return;
- if (!ls[u]) ls[u] = ++siz;
- if (!rs[u]) rs[u] = ++siz;
- lazy[ls[u]] += lazy[u]; lazy[rs[u]] += lazy[u];
- int mid = l + r >> 1;
- sum[ls[u]] += lazy[u] * (mid - l + 1);
- sum[rs[u]] += lazy[u] * (r - mid);
- lazy[u] = 0;
- }
- void modify(int& u,int l,int r){
- if (!u) u = ++siz;
- if (l >= L && r <= R){
- sum[u] += (r - l + 1);
- lazy[u]++;
- }else {
- pd(u,l,r);
- int mid = l + r >> 1;
- if (mid >= L) modify(ls[u],l,mid);
- if (mid < R) modify(rs[u],mid + 1,r);
- sum[u] = sum[ls[u]] + sum[rs[u]];
- }
- }
- int Query(int u,int l,int r){
- if (!u) return 0;
- if (l >= L && r <= R) return sum[u];
- else {
- pd(u,l,r);
- int mid = l + r >> 1;
- if (mid >= R) return Query(ls[u],l,mid);
- else if (mid < L) return Query(rs[u],mid + 1,r);
- else return Query(ls[u],l,mid) + Query(rs[u],mid + 1,r);
- }
- }
- inline void insert(){
- int u = 1,l = 1,r = N;
- L = a; R = b;
- while (l < r){
- int mid = l + r >> 1;
- modify(rt[u],1,n);
- if (c <= mid) r = mid,u = u << 1;
- else l = mid + 1,u = u << 1 | 1;
- }
- modify(rt[u],1,n);
- }
- inline void solve(){
- int u = 1,l = 1,r = N,t;
- L = a; R = b;
- while (l < r){
- int mid = l + r >> 1;
- t = Query(rt[u << 1],1,n);
- if (c <= t) r = mid,u = u << 1;
- else l = mid + 1,u = u << 1 | 1,c -= t;
- }
- printf("%d\n",N - l + 1 - n);
- }
- int main()
- {
- int cmd;
- n = read(); m = read();N = n * 2 + 1;
- while (m--){
- cmd = read(); a = read(); b = read(); c = read();
- if (cmd & 1){
- c += n;
- c = N - c + 1;
- insert();
- }else solve();
- }
- return 0;
- }
- #include<iostream>
- #include<cstdio>
- #include<cstring>
- #include<algorithm>
- #define LL long long int
- #define REP(i,n) for (int i = 1; i <= (n); i++)
- #define fo(i,x,y) for (int i = (x); i <= (y); i++)
- #define Redge(u) for (int k = head[u]; k != -1; k = edge[k].next)
- using namespace std;
- const int maxn = 500005,maxm = 100005,INF = 1000000000;
- //begin 18:00 end 19:27
- inline int read(){
- int out = 0,flag = 1;char c = getchar();
- while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
- while (c >= 48 && c <= 57) {out = out * 10 + c - 48; c = getchar();}
- return out * flag;
- }
- int n,M;
- struct node{
- int l,r,v,p,id,k;
- }e[maxn];
- inline bool cmp(const node& a,const node& b){
- return a.k == b.k ? a.id < b.id : a.k < b.k;
- }
- int ans[maxn];
- int sum[4 * maxn],lazy[4 * maxn],L,R;
- bool cl[4 * maxn];
- inline void pd(int u,int l,int r){
- if (cl[u]) sum[u<<1] = sum[u<<1|1] = lazy[u<<1] = lazy[u<<1|1] = cl[u] = 0,cl[u<<1] = cl[u<<1|1] = true;
- if (lazy[u]){
- int mid = l + r >> 1;
- sum[u<<1] += (mid - l + 1) * lazy[u];
- sum[u<<1|1] += (r - mid) * lazy[u];
- lazy[u<<1] += lazy[u];
- lazy[u<<1|1] += lazy[u];
- lazy[u] = 0;
- }
- }
- inline void update(int u,int l,int r){
- pd(u,l,r);
- if (l >= L && r <= R){
- sum[u] += (r - l + 1);
- lazy[u]++;
- }
- else {
- int mid = l + r >> 1;
- if (mid >= L) update(u<<1,l,mid);
- if (mid < R) update(u<<1|1,mid + 1,r);
- sum[u] = sum[u<<1] + sum[u<<1|1];
- }
- }
- inline int Query(int u,int l,int r){
- pd(u,l,r);
- if (l >= L && r <= R) return sum[u];
- else {
- int mid = l + r >> 1;
- if (mid >= R) return Query(u<<1,l,mid);
- else if (mid < L) return Query(u<<1|1,mid + 1,r);
- else return Query(u<<1,l,mid) + Query(u<<1|1,mid + 1,r);
- }
- }
- void solve(int l,int r,int el,int er){
- if (el > er) return;
- if (l == r){
- for (int i = el; i <= er; i++)
- if (e[i].p == 2) ans[e[i].id] = l;
- }
- else {
- cl[1] = true; sum[1] = lazy[1] = 0;
- int mid = (l + r) >> 1,i = el - 1,t;
- for (int k = el; k <= er; k++){
- if (e[k].p == 1){
- if (e[k].v > mid){
- L = e[k].l; R = e[k].r;
- update(1,1,n);
- e[k].k = 1;
- }else {
- e[k].k = 0;
- i++;
- }
- }
- else {
- L = e[k].l; R = e[k].r;
- t = Query(1,1,n);
- if (e[k].v <= t) e[k].k = 1;
- else {
- e[k].v -= t;
- e[k].k = 0;
- i++;
- }
- }
- }
- sort(e + el,e + er + 1,cmp);
- solve(l,mid,el,i);
- solve(mid + 1,r,i + 1,er);
- }
- }
- void init(){
- n = read(); M = read();
- REP(i,M){
- e[i].p = read();
- e[i].l = read();
- e[i].r = read();
- e[i].v = read();
- if (e[i].p == 1) e[i].v = e[i].v + n;
- e[i].id = i;
- }
- memset(ans,-1,sizeof(ans));
- }
- void print(){
- for (int i = 1; i <= M; i++)
- if (ans[i] != -1) {
- if (ans[i] == n) printf("50000\n");
- else printf("%d\n",ans[i] - n);
- }
- }
- int main()
- {
- init();
- solve(0,2 * n,1,M);
- print();
- return 0;
- }
BZOJ3110 K大数查询 【线段树 + 整体二分 或 树套树(非正解)】的更多相关文章
- bzoj3110 [Zjoi2013]K大数查询——线段树套线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 外层权值线段树套内层区间线段树: 之所以外层权值内层区间,是因为区间线段树需要标记下传 ...
- bzoj 3110 [Zjoi2013]K大数查询——线段树套线段树(标记永久化)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 第一道线段树套线段树! 第一道标记永久化! 为什么为什么写了两个半小时啊…… 本想线段 ...
- 【BZOJ-3110】K大数查询 整体二分 + 线段树
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 6265 Solved: 2060[Submit][Sta ...
- BZOJ3110[Zjoi2013]K大数查询(树状数组+整体二分)
3110 [Zjoi2013]K大数查询 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位置加入一个数c如果是2 a b c形式,表示询问从第a ...
- 【BZOJ3110】K大数查询(整体二分)
[BZOJ3110]K大数查询(整体二分) 题面 BZOJ 题解 看了很久整体二分 一直不知道哪里写错了 ... 又把树状数组当成线段树区间加法来用了.. 整体二分还是要想清楚在干什么: 我们考虑第\ ...
- BZOJ_3110_[Zjoi2013]K大数查询_整体二分+树状数组
BZOJ_3110_[Zjoi2013]K大数查询_整体二分+树状数组 Description 有N个位置,M个操作.操作有两种,每次操作如果是1 a b c的形式表示在第a个位置到第b个位置,每个位 ...
- BZOJ 3110 [Zjoi2013]K大数查询(整体二分)
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 11654 Solved: 3505[Submit][St ...
- bzoj 3110: [Zjoi2013]K大数查询 树状数组套线段树
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 1384 Solved: 629[Submit][Stat ...
- BZOJ 3110([Zjoi2013]K大数查询-区间第k大[段修改,在线]-树状数组套函数式线段树)
3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec Memory Limit: 512 MB Submit: 418 Solved: 235 [ Submit][ ...
随机推荐
- JS基础,课堂作业,成绩练习
成绩练习 <script> var name = prompt("请输入学生姓名:"); var degree = parseInt(prompt("请输入学 ...
- Android 测试 之MonkeyRunner
一.什么是MonkeyRunner monkeyrunner工具提供了一个API,使用此API写出的程序可以在Android代码之外控制Android设备和模拟器.通过monkeyrunner,您可以 ...
- ip route ifconfig 基本命令
1.route命令 route –n route add –net 192.168.2.0/24 dev eth0 route add –net 192.168.2.0 netmask 255.255 ...
- day06 再谈编码 and 作业讲解
1. 小数据池,(其他语言又叫常量池) id() 查看变量的内存地址 is和== is 判断内存地址是否一致 == 判断内容是否一致 小数据池的作用: 为了快速的创建字符串对象, 可以减少内存的浪费 ...
- 软银集团和共享办公空间公司WeWork在日本成立合资公司
[TechWeb报道]7月18日消息,据国外媒体报道,软银集团和共享办公空间公司WeWork联合宣布,在日本成立合资公司WeWork Japan. 该合资公司将在日本开设联合办公空间,于明年初在东京设 ...
- Python 装饰器Decorator(一)
(一) 装饰器基础知识 什么是Python装饰器?Python里装饰器是一个可调用的对象(函数),其参数是另一个函数(被装饰的函数) 假如有一个名字为somedecorator的装饰器,target是 ...
- zend安装及破解
Zend下载 https://pan.baidu.com/s/1fCfUQ0j7dxEtOzbNnsgODg 破解: 1.打开dmg文件安装,将Zend Studio拖拽至applications进行 ...
- Arctic Network POJ 2349 (最小生成树思想)
Description The Department of National Defence (DND) wishes to connect several northern outposts by ...
- C++:const_cast的简单理解
前言:const_cast是我比较头疼的一个知识点,最近查阅了很多资料,也翻看了很多他人的博客,故在此将自己目前学习到的有关const_cast知识做一个简单的总结 一.什么是const_cast 简 ...
- pandas中DataFrame的ix,loc,iloc索引方式的异同
pandas中DataFrame的ix,loc,iloc索引方式的异同 1.loc: 按照标签索引,范围包括start和end 2.iloc: 在位置上进行索引,不包括end 3.ix: 先在inde ...