tree(二维偏序)

最近接触到一些偏序的东西。

传统线段树非叶子节点的划分点mid=(l+r)/2,但小R线段树mid是自己定的。但满足l<=mid<r,其余条件同原来线段树。那么不难发现如下性质:1.该线段树的节点个数依然为2N-1.2.该线段树深度可能会超过O(logn)。3.该线段树区间定位所包含的线段树节点个数可能超过O(logn)。但区间定位的结果依然是唯一的。小R给你这样一个小R线段树,每次询问给定区间的区间定位个数。

​ 这道题n和询问个数都到了1e5,所以考虑nlogn的做法。

​ 我们发现区间定位个数(答案)和完全被该区间包含的节点个数所相关。具体性质如下:区间定位个数(答案) = 2 * 区间长度 - 完全被该区间包含的节点个数。不难发现完全被该区间包含的节点个可以看作一个森林,而区间定位出的那些线段节点可以看作这些森林当中数的根。由于一个长度为l的根的树的节点个数为 2 ∗ l − 1,那么这个森林当中每有一棵树,就会对“2 * 区间长度”这个总和上减去一,故满足性质。那么我们将问题转化为求完全被该区间包含的节点个数。

​ 那么现在问题就是在所有的区间中,找到[l,r]所完全包含的区间。这是个二维偏序的问题。个人理解所谓二维偏序就是询问许多向量中,比询问的向量小的向量。由于这是偏序关系,所以不是任意两个向量都可以比较大小。这里的定义是如果一个区间l,r,l<=l2,并且r>=r2,那么l,r小于l2,r2。具体做法就是一维排序,然后i从大到小(从小到大也可以,只不过这里dfs出来顺序是从小到大的,那么就从大到小扫一遍),对于左端点在当前询问左端点右侧的区间,把其右侧端点在树状数组里mark一下。然后查询的时候,只查询1到r被mark的值。这样由于只统计了左端点在l右侧的区间,并且区间的右端点被限制在1到r以内,所以所有区间都会被扫到。

​ 感觉这种方法真奇妙。。

#include <cstdio>
#include <algorithm>
using namespace std; const int maxn=4e5+5;
int n, m, cntseg, cntline, cntq, beg;
int atree[maxn]; struct Node{
int lson, rson, mid;
}node[maxn]; struct Line{
int l, r, id, ans;
void set(const int &x, const int &y){ l=x, r=y; }
}line[maxn], queries[maxn];
bool cmp1(const Line &x, const Line &y){
return x.l==y.l?x.r<y.r:x.l<y.l;
}
bool cmp2(const Line &x, const Line &y){
return x.id<y.id;
} void build(int &now, int l, int r){
line[cntline++].set(l, r);
if (l==r) return;
if (!now) now=++cntseg;
scanf("%d", &node[now].mid);
build(node[now].lson, l, node[now].mid);
build(node[now].rson, node[now].mid+1, r);
} inline int lowbit(int x){ return x&(-x); } void modify(int now){
while (now<=n){
++atree[now];
now+=lowbit(now);
}
} int query(int now){
int re=0;
while (now){
re+=atree[now];
now-=lowbit(now);
} return re;
} int main(){
scanf("%d%d", &n, &m);
build(beg, 1, n);
int l, r;
for (int i=0; i<m; ++i){
scanf("%d%d", &l, &r);
queries[cntq].set(l, r);
queries[cntq++].id=i;
}
sort(queries, queries+m, cmp1);
int j=cntline-1;
for (int i=m-1; i>=0; --i){
//如果这个区间,它的左端点在查询的左端点右边
//说明它对查询的左端点是有效果的
while (line[j].l>=queries[i].l){
modify(line[j].r); --j;
}
queries[i].ans=2*(queries[i].r-queries[i].l+1)-query(queries[i].r);
}
sort(queries, queries+m, cmp2);
for (int i=0; i<m; ++i)
printf("%d\n", queries[i].ans);
return 0;
}

二维偏序 tree的更多相关文章

  1. 洛谷 P1972 [SDOI2009]HH的项链-二维偏序+树状数组+读入挂(离线处理,思维,直接1~n一边插入一边查询),hahahahahahaha~

    P1972 [SDOI2009]HH的项链 题目背景 无 题目描述 HH 有一串由各种漂亮的贝壳组成的项链.HH 相信不同的贝壳会带来好运,所以每次散步完后,他都会随意取出一段贝壳,思考它们所表达的含 ...

  2. [luogu4479][BJWC2018]第k大斜率【二维偏序+二分+离散化+树状数组】

    传送门 https://www.luogu.org/problemnew/show/P4479 题目描述 在平面直角坐标系上,有 n 个不同的点.任意两个不同的点确定了一条直线.请求出所有斜率存在的直 ...

  3. 【坐标变换】【二维偏序】【线段树】Gym - 100820G - Racing Gems

    题意:第一象限有n个点,你从x正半轴任选一个位置出发,vy恒定,vx可以任意变化,不过只能在-vy/r到vy/r之间变化,问你最多能经过多少个点. 暴力dp是n^2,不可取. 注意到,一个点,所能到达 ...

  4. cdq分治入门学习 cogs 1752 Mokia nwerc 2015-2016 G 二维偏序

    /* CDQ分治的对象是时间. 即对于一个时间段[L, R],我们取mid = (L + R) / 2. 分治的每层只考虑mid之前的修改对mid之后的查询的贡献,然后递归到[L,mid],(mid, ...

  5. 二维偏序+树状数组【P3431】[POI2005]AUT-The Bus

    Description Byte City 的街道形成了一个标准的棋盘网络 – 他们要么是北南走向要么就是西东走向. 北南走向的路口从 1 到 n编号, 西东走向的路从1 到 m编号. 每个路口用两个 ...

  6. 【贪心】【二维偏序】【权值分块】bzoj1691 [Usaco2007 Dec]挑剔的美食家

    既然题目中的要求满足二维偏序,那么我们很自然地想到将所有东西(草和牛)都读进来之后,对一维(美味度)排序,然后在另一维(价值)中取当前最小的. 于是,Splay.mutiset.权值分块什么的都支持查 ...

  7. 【二维偏序】【树状数组】【权值分块】【分块】poj2352 Stars

    经典问题:二维偏序.给定平面中的n个点,求每个点左下方的点的个数. 因为 所有点已经以y为第一关键字,x为第二关键字排好序,所以我们按读入顺序处理,仅仅需要计算x坐标小于<=某个点的点有多少个就 ...

  8. 树状数组 二维偏序【洛谷P3431】 [POI2005]AUT-The Bus

    P3431 [POI2005]AUT-The Bus Byte City 的街道形成了一个标准的棋盘网络 – 他们要么是北南走向要么就是西东走向. 北南走向的路口从 1 到 n编号, 西东走向的路从1 ...

  9. 【BZOJ1109】[POI2007]堆积木Klo 二维偏序

    [BZOJ1109][POI2007]堆积木Klo Description Mary在她的生日礼物中有一些积木.那些积木都是相同大小的立方体.每个积木上面都有一个数.Mary用他的所有积木垒了一个高塔 ...

随机推荐

  1. 尴尬,qt出现错误,然后莫名又好了..

    Starting D:\code\qt\myChess\day01\build-HelloQT-Desktop_Qt_5_5_0_MinGW_32bit-Debug\debug\HelloQT.exe ...

  2. 【leetcode刷题笔记】Valid Palindrome

    Given a string, determine if it is a palindrome, considering only alphanumeric characters and ignori ...

  3. python 3 serial module install

    /************************************************************************* * python 3 serial module ...

  4. [原]NYOJ-组合数-32

    大学生程序代写 http://acm.nyist.net/JudgeOnline/problem.php?pid=32 /*组合数 时间限制:3000 ms  |  内存限制:65535 KB 难度: ...

  5. google android sdk下载hoosts

    203.208.46.146 www.google.com #这行是为了方便打开Android开发官网 现在好像不VPN也可以打开 74.125.113.121 developer.android.c ...

  6. BZOJ3033 太鼓达人

    3033: 太鼓达人 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 690  Solved: 497[Submit][Status][Discuss] ...

  7. bzoj 2850 巧克力王国——KDtree

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2850 改一下估价即可.判断子树能否整个取或者是否整个不能取,时间好像就能行了? 因为有负数, ...

  8. Poj 1316 Self Numbers(水题)

    一.Description In 1949 the Indian mathematician D.R. Kaprekar discovered a class of numbers called se ...

  9. poj 2388 Who's in the Middle(快速排序求中位数)

    一.Description FJ is surveying his herd to find the most average cow. He wants to know how much milk ...

  10. 机器学习:k-NN算法(也叫k近邻算法)

    一.kNN算法基础 # kNN:k-Nearest Neighboors # 多用于解决分裂问题 1)特点: 是机器学习中唯一一个不需要训练过程的算法,可以别认为是没有模型的算法,也可以认为训练数据集 ...