luogu3834 【模板】可持久化线段树1(主席树)
关键字:线段树 可持久化
线段树:当版本(即对应的原序列的区间[1,r])一定时,每个节点的left,right下标为值域,值为其对应的原序列区间[1,r]中元素大小在值域中的元素个数。
可持久化:新版本(对应原序列[1,r])在旧版本(对应原序列[1,r-1])上建立,从树根向树叶构造,在旧版本的节点的旁边构造新版本的节点,值为旧版本节点值+1。搜索新版本树时,从新版本树根开始搜索即可。求区间第k大,同时遍历【1,l-1】对应版本树和【1,r】对应版本树,通过节点值的差来得到第k个节点。
#include <cstdio>
#include <cstring>
#include <cassert>
#include <cmath>
#include <algorithm>
using namespace std; #define LOOP(i,n) for(int i=1; i<=n; i++)
#define INF 0x3f3f3f3f
const int MAX_ROOT=2e5;
int Seq[MAX_ROOT]; struct Node
{
int Cnt;
Node *LeftSon, *RightSon;
Node()
{
Cnt = ;
LeftSon = RightSon = NULL;
}
}; Node *_roots[MAX_ROOT];
int _vCount;
int MinL, MaxR; void Init(int l, int r)
{
_vCount = ;
MinL = l;
MaxR = r;
_roots[] = new Node();
} Node *Update(Node *prev, Node *&cur, int l, int r, int p)
{
cur = new Node();
*cur = *prev;
cur->Cnt++;
if (l == r)
return cur;
int mid = (r - l) / + l;
if (p <= mid)
{
if (!prev->LeftSon)
prev->LeftSon = new Node();
Update(prev->LeftSon, cur->LeftSon, l, mid, p);
}
else
{
if (!prev->RightSon)
prev->RightSon = new Node();
Update(prev->RightSon, cur->RightSon, mid + , r, p);
}
return cur;
} int Query(Node *prev, Node *cur, int l, int r, int k)
{
if (l == r)
return l;
int prevLeft = prev->LeftSon ? prev->LeftSon->Cnt : , curLeft = cur->LeftSon ? cur->LeftSon->Cnt : ;
int leftCnt = curLeft - prevLeft;
int mid = (r - l) / + l;
if (k <= leftCnt)
{
if (!prev->LeftSon)
prev->LeftSon = new Node();
return Query(prev->LeftSon, cur->LeftSon, l, mid, k);
}
else
{
if (!prev->RightSon)
prev->RightSon = new Node();
return Query(prev->RightSon, cur->RightSon, mid + , r, k - leftCnt);
}
} void Update(int r)
{
Update(_roots[r - ], _roots[r], MinL, MaxR, Seq[r]);
} int Query(int l, int r, int k)
{
return Query(_roots[l - ], _roots[r], MinL, MaxR, k);
} int main()
{
#ifdef _DEBUG
freopen("c:\\noi\\source\\input.txt", "r", stdin);
#endif
int seqCnt, queryCnt, l, r, k, minL = INF, maxR = -INF;
scanf("%d%d", &seqCnt, &queryCnt);
LOOP(i, seqCnt)
{
scanf("%d", i + Seq);
minL = min(Seq[i], minL);
maxR = max(Seq[i], maxR);
}
Init(minL, maxR);
LOOP(i, seqCnt)
Update(i);
LOOP(i, queryCnt)
{
scanf("%d%d%d", &l, &r, &k);
printf("%d\n", Query(l, r, k));
}
return ;
}
luogu3834 【模板】可持久化线段树1(主席树)的更多相关文章
- P3919 【模板】可持久化数组 -初步探究主席树
本篇blog主要是给自己(大家)看的. 感谢longlongzhu123奆佬(此人初二LCT)的指点,使本蒟蒻可以快速开始主席树入门. what is 主席树? $ $主席树这个名字只不 ...
- 归并树 划分树 可持久化线段树(主席树) 入门题 hdu 2665
如果题目给出1e5的数据范围,,以前只会用n*log(n)的方法去想 今天学了一下两三种n*n*log(n)的数据结构 他们就是大名鼎鼎的 归并树 划分树 主席树,,,, 首先来说两个问题,,区间第k ...
- POJ 2104 K-th Number(分桶,线段树,主席树)
一道比较经典的数据结构题.可以用多种方式来做. 一,分桶法(平方分解). 根据数字x的大小和区间内不大于x的数字数量cnt的单调性,可知第k大数kth对应的cnt应该满足cnt≥k, 且kth是满足条 ...
- 【题解】BZOJ3489 A Hard RMQ problem(主席树套主席树)
[题解]A simple RMQ problem 占坑,免得咕咕咕了,争取在2h内写出代码 upd:由于博主太菜而且硬是要用指针写两个主席树,所以延后2hQAQ upd:由于博主太菜而且太懒所以他决定 ...
- poj 2104 K-th Number 划分树,主席树讲解
K-th Number Input The first line of the input file contains n --- the size of the array, and m --- t ...
- 【BZOJ4771】七彩树(主席树)
[BZOJ4771]七彩树(主席树) 题面 BZOJ 题解 如果没有深度限制,每次只询问子树内的颜色个数,除了树套树\(dfs\)序加前驱或者后继强行二维数点之外,还有这样一种做法: 把所有相同颜色的 ...
- 洛谷P3248 树 [HNOI2016] 主席树+倍增+分治
正解:主席树+倍增+分治 解题报告: 传送门! 首先看到这题会想到之前考过的这题 但是那题其实简单一些,,,因为那题只要用个分治+预处理就好,只是有点儿思维难度而已 这题就不一样,因为它说了是按照原树 ...
- BZOJ_2588_Spoj 10628. Count on a tree_树剖+主席树
BZOJ_2588_Spoj 10628. Count on a tree_树剖+主席树 题意: 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastan ...
- POJ 2761 Feed the dogs(平衡树or划分树or主席树)
Description Wind loves pretty dogs very much, and she has n pet dogs. So Jiajia has to feed the dogs ...
- 【洛谷P3834】(模板)可持久化线段树 1(主席树)
[模板]可持久化线段树 1(主席树) https://www.luogu.org/problemnew/show/P3834 主席树支持历史查询,空间复杂度为O(nlogn),需要动态开点 本题用一个 ...
随机推荐
- ajax不跳转页面的快速删除操作,可添加美观样式
以前我们讲的删除是利用嵌入php代码,跳转到另一个页面,从而降低了删除速度,但我们今天讲的利用ajax不仅可以达到不跳页面快速删除,并且能添加特效来美化页面. 上代码,我们先来做主页面 <!DO ...
- 01--TCP状态转换
参考大牛文章: http://www.cnblogs.com/qlee/archive/2011/07/12/2104089.html
- 我的 Windows 10 的基本配置
Windows 10 的基本配置 功能性 开启 .Net Framework 3.5(包括 .NET 2.0 和 3.0) 旧版本 Windows 10 默认只安装了 .Net Framework 4 ...
- C# 把时间 月 //把第一个0替换为空
string str = "2019-01"; //name: "2019-01月" str = str.Substring(str.Length - , ); ...
- day02-操作系统、编程语言分类及python安装
目录 操作系统 编程语言分类 安装python解释器 操作系统 操作系统有什么用 操作系统能接受外部指令转化成0和1,并把一些对硬件的复杂操作简化成一个个简单的接口,作为中间人连接硬件和软件 计算机三 ...
- vue.js---利用vue cli脚手架工具+webpack创建项目遇到的坑
1.Eslint js代码规范报错 WARNING Compiled with 2 warnings 10:43:26 ✘ http://eslint.org/docs/rules/quotes St ...
- vue移动端地址三级联动组件(一)
vue移动端地区三级联动 省,市,县.用的vue+mintUi 因为多级联动以及地区的规则比较多.正好有时间自己写了一个.有问题以及建议欢迎指出.涉及到dom移动,所以依赖vue+jquery.这边数 ...
- Chat Group gym101775A(逆元,组合数)
传送门:Chat Group(gym101775A) 题意:一个宿舍中又n个人,最少k(k >= 3)个人就可以建一个讨论组,问最多可以建多少个不同的讨论组. 思路:求组合数的和,因为涉及除法取 ...
- Google Cloud SSH 连接配置
设置当前用户及root用户的密码 sudo passwd xx-user # 输入新密码 sudo passwd root # 输入新密码(建议保持一样) 在本地生成私钥和公钥 cd ~/.ssh s ...
- Windows Server 2008安装教程
系统简介 windows server 2008是迄今为止最灵活.最稳定的windows 操作系统.Windows server 2008 的安装过程是基于镜像文件的,主要版本:Windows Ser ...