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

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

Sample Output

1
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

Source

Solution

  树套树。。。说难不难,说简单不简单。曾经花了一星期理解内外线段树的关系。。。一直不理解怎么“套”一棵树的

  树套树的一般做法是用两个树形结构,外层代表区间,内层代表该区间下的权值信息或其它的什么

  相当于一个二维矩阵,一维是区间信息一维是权值信息。

  嘛,其实就是外层线段树纪录它对应的内层线段树的节点编号,内层线段树纪录该区间下与某权值范围的的结点信息

  为了省空间,需用动态开点的姿势。在这里给各位提个醒:这道题开2000000就好。

  嗯= =其实就是把线段树update和query写两遍233

  蛋碎了一地。数组开大MLE,数组开小RE。

  这道题外层区间内层权值有点麻烦,所以换一种思路:外层权值线段树内层区间线段树。递归查询某一段权值范围内有多少数,若个数小于询问数则向左子树递归,否则向右子树递归。

  = =

  好吧我的确给别人讲不懂 _(:з」∠)_

  代码常熟大地飞起。标记永久化是什么可以吃吗

  yky大爷讲了一种奥妙丛丛的压缩空间的方法:把结构体里的变量压缩为x位整型。因为好像存和要用unsigned int于是干脆用long long。具体用法看代码。

 #include <bits/stdc++.h>
using namespace std;
typedef long long ll;
struct seg1
{
ll l:, r:, sum:, lazy:;
}p1[];
struct operation
{
ll op, l, r, val;
}op[];
ll n, p2[], ptot, cd[], ctot, ql, qr, val; void push_up(ll o, ll l, ll r)
{
p1[o].sum = p1[p1[o].l].sum + p1[p1[o].r].sum + p1[o].lazy * (r - l + );
} void push_down(ll o, ll l, ll r)
{
ll mid = (l + r) >> ;
if(!p1[o].l) p1[o].l = ++ptot;
if(!p1[o].r) p1[o].r = ++ptot;
if(p1[o].lazy)
{
p1[p1[o].l].lazy += p1[o].lazy;
p1[p1[o].r].lazy += p1[o].lazy;
p1[p1[o].l].sum += p1[o].lazy * (mid - l + );
p1[p1[o].r].sum += p1[o].lazy * (r - mid);
p1[o].lazy = ;
}
} void update1(ll o, ll l, ll r)
{
ll mid = (l + r) >> ;
if(ql <= l && r <= qr)
{
p1[o].sum += r - l + , p1[o].lazy++;
return;
}
push_down(o, l, r);
if(ql <= mid) update1(p1[o].l, l, mid);
if(mid < qr) update1(p1[o].r, mid + , r);
push_up(o, l, r);
} void update2(ll o, ll l, ll r)
{
ll mid = (l + r) >> ;
if(!p2[o]) p2[o] = ++ptot;
update1(p2[o], , n);
if(l == r) return;
if(val <= mid) update2(o << , l, mid);
else update2(o << | , mid + , r);
} ll query1(ll o, ll l, ll r)
{
ll mid = (l + r) >> , cnt = ;
if(ql <= l && r <= qr) return p1[o].sum;
push_down(o, l, r);
if(ql <= mid) cnt = query1(p1[o].l, l, mid);
if(mid < qr) cnt += query1(p1[o].r, mid + , r);
return cnt;
} ll query2(ll o, ll l, ll r, ll rk)
{
ll mid = (l + r) >> ;
if(l == r) return l;
if(!p2[o]) p2[o] = ++ptot;
val = query1(p2[o << | ], , n);
if(val < rk) return query2(o << , l, mid, rk - val);
return query2(o << | , mid + , r, rk);
} int main()
{
ll m;
scanf("%lld%lld", &n, &m);
for(ll i = ; i <= m; i++)
{
scanf("%lld%lld%lld%lld", &op[i].op, &op[i].l, &op[i].r, &op[i].val);
if(op[i].op == ) cd[++ctot] = op[i].val;
}
sort(cd + , cd + ctot + );
for(int i = ; i <= m; i++)
if(op[i].op == )
op[i].val = lower_bound(cd + , cd + ctot + , op[i].val) - cd;
for(ll i = ; i <= m; i++)
if(op[i].op == )
{
ql = op[i].l, qr = op[i].r, val = op[i].val;
update2(, , ctot);
}
else
{
ql = op[i].l, qr = op[i].r;
printf("%lld\n", cd[query2(, , ctot, op[i].val)]);
}
return ;
}

[BZOJ3110] [Zjoi2013] K大数查询 (树套树)的更多相关文章

  1. P3332 [ZJOI2013]K大数查询(线段树套线段树+标记永久化)

    P3332 [ZJOI2013]K大数查询 权值线段树套区间线段树 把插入的值离散化一下开个线段树 蓝后每个节点开个线段树,维护一下每个数出现的区间和次数 为了防止MLE动态开点就好辣 重点是标记永久 ...

  2. BZOJ3110[Zjoi2013]K大数查询(树状数组+整体二分)

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

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

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

  4. BZOJ3110[Zjoi2013]K大数查询——权值线段树套线段树

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

  5. bzoj3110: [Zjoi2013]K大数查询 【cdq分治&树套树】

    模板题,折腾了许久. cqd分治整体二分,感觉像是把询问分到答案上. #include <bits/stdc++.h> #define rep(i, a, b) for (int i = ...

  6. 【bzoj3110】[Zjoi2013]K大数查询 整体二分+树状数组区间修改

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

  7. BZOJ3110: [Zjoi2013]K大数查询

    喜闻乐见的简单树套树= =第一维按权值建树状数组,第二维按下标建动态开点线段树,修改相当于第二维区间加,查询在树状数组上二分,比一般的线段树还短= =可惜并不能跑过整体二分= =另外bzoj上的数据有 ...

  8. bzoj3110: [Zjoi2013]K大数查询 【树套树,标记永久化】

    //========================== 蒟蒻Macaulish:http://www.cnblogs.com/Macaulish/  转载要声明! //=============== ...

  9. bzoj3110 [Zjoi2013]K大数查询——线段树套线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3110 外层权值线段树套内层区间线段树: 之所以外层权值内层区间,是因为区间线段树需要标记下传 ...

随机推荐

  1. 像我这样优雅地进行Spring整合MongoDB

    本文重点是要将mongodb与spring整合到项目中去,在实践中发现问题,追踪问题,然后解决问题. 一.准备 Maven.Spring(spring-data-mongodb) spring Dat ...

  2. HTTPS的原理解析

    http://www.cnblogs.com/alisecurity/p/5939336.html 外加文档

  3. 登录功能(MD5加密)

    登录这个功能,是不管哪个项目都会用到的,登录做的好坏,安全性的保障将直接影响到整个系统的成败,尤其是一些安全性要求比较严格的项目 1.首先需要对密码进行加密,这里用到的是md5加密,需要在login. ...

  4. django-高级视图和url配置

    高级视图和url配置 一.URLconf技巧 1.流线型化函数导入 对于配置url,我们可以使用以下几种方式: (1)引入view中的函数 from firstSite.view import cur ...

  5. 微博爬虫“免登录”技巧详解及 Java 实现(业余草的博客)

    一.微博一定要登录才能抓取? 目前,对于微博的爬虫,大部分是基于模拟微博账号登录的方式实现的,这种方式如果真的运营起来,实际上是一件非常头疼痛苦的事,你可能每天都过得提心吊胆,生怕新浪爸爸把你的那些账 ...

  6. 一步一步带你实现virtual dom(一)

    一步一步带你实现virtual dom(一) 一步一步带你实现virtual dom(二)--Props和事件 要写你自己的虚拟DOM,有两件事你必须知道.你甚至都不用翻看React的源代码,或者其他 ...

  7. 浅谈CDN、SEO、XSS、CSRF

    CDN 什么是CDN 初学Web开发的时候,多多少少都会听过这个名词->CDN. CDN在我没接触之前,它给我的印象是用来优化网络请求的,我第一次用到CDN的时候是在找JS文件时.当时找不到相对 ...

  8. 64位Kali无法顺利执行pwn1问题的解决方案

    问题描述 ​ 环境:VMware Fusion + kali-linux-2018.1-amd64.iso ​ 问题:在Terminal利用./pwn1执行pwn1会出现 bash: ./pwn1:没 ...

  9. postman模拟HttpPost请求的方法

    开始想装postman的Google浏览器插件的,但是发现应用商店无法搜索,下载的拖进扩展也装不上... 于是找到了这个绿色版的Postman桌面程序!有需要的可以下载,点击下载:http://dow ...

  10. 使用java实现阿里云消息队列简单封装

    一.前言 最近公司有使用阿里云消息队列的需求,为了更加方便使用,本人用了几天时间将消息队列封装成api调用方式以方便内部系统的调用,现在已经完成,特此记录其中过程和使用到的相关技术,与君共勉. 现在阿 ...