题目传送门:https://ac.nowcoder.com/acm/contest/1107/C

题意:给出一个区间,求最大的 h ,使得区间内至少有 h 个数 大于等于 h.

思路:1.需要区间有序,那么就需要使用 主席树。

   2.二分答案。

    2.1 —— 一开始我的思路是直接对每一个查询二分答案 h。

        然后判断第 len(区间个数)-h+1 小是否大于等于 h,这样二分得出最大h,但是T了。

        一开始以为是卡常了,于是各种 骚操作 还是没有什么卵用。

        于是在一次优化 “ 由于n小可以去掉离散化” ,依然T了之后。忽然发现可以直接二分位置,也就是直接在主席树查询中直接找到答案。

    2.2 —— 在查询过程中,我们就不断的二分位置,判定条件是:

        主席树中,

        h右边的个数是否大于h,如果是,那么就往右子树走,否则就往左子树走。

        记得不要离散化,因为我们二分的是位置,而不是大小。

 

//#pragma comment(linker, "/STACK:1024000000,1024000000")
//#pragma GCC optimize(2)
#include <bits/stdc++.h> using namespace std;
typedef double dou;
typedef long long ll;
typedef pair<int, int> pii;
typedef map<int, int> mii; #define pai acos(-1.0)
#define M 4000005
#define inf 0x3f3f3f3f
#define mod 1000000007
#define IN inline
#define W(a) while(a)
#define lowbit(a) a&(-a)
#define left k<<1
#define right k<<1|1
#define lson L, mid, left
#define rson mid + 1, R, right
#define ms(a,b) memset(a,b,sizeof(a))
#define Abs(a) (a ^ (a >> 31)) - (a >> 31)
#define random(a,b) (rand()%(b+1-a)+a)
#define false_stdio ios::sync_with_stdio(false),cin.tie(0),cout.tie(0) int n, cnt, Q;
int num[M];
int T[M], Ls[M], Rs[M], sum[M]; IN int Built(int L, int R) {
int rt = ++cnt;
if (L < R) {
int mid = (L + R) >> ;
Ls[rt] = Built(L, mid);
Rs[rt] = Built(mid + , R);
}
return rt;
} IN int updata(int L, int R, int pre, int id) {
int rt = ++cnt;
Ls[rt] = Ls[pre], Rs[rt] = Rs[pre], sum[rt] = sum[pre] + ;
if (L < R) {
int mid = (L + R) >> ;
if (id <= mid)Ls[rt] = updata(L, mid, Ls[pre], id);
else Rs[rt] = updata(mid + , R, Rs[pre], id);
}
return rt;
} //Rtot是上一个节点的右子节点的个数
IN int query(int u, int v, int L, int R, int Rtot) {
if (L == R)return L;
int R_sum = sum[Rs[v]] - sum[Rs[u]];//右子节点的个数
int mid = (L + R) >> ;
//判定条件
if (mid - Rtot + > R_sum)return query(Ls[u], Ls[v], L, mid, Rtot + R_sum);
else return query(Rs[u], Rs[v], mid + , R, Rtot);
} IN int read() {//读入挂
int x = ; bool f = ; char ch = getchar();
while (ch < '' || '' < ch)
f |= ch == '-', ch = getchar();
while ('' <= ch && ch <= '')
x = x * + ch - '', ch = getchar();
return f ? -x : x;
} int main() {
W(scanf("%d%d", &n, &Q) != EOF) {
cnt = ;
T[] = Built(, n);
for (register int i = ; i <= n; i++)num[i] = read();
for (register int i = ; i <= n; i++)T[i] = updata(, n, T[i - ], num[i]); int l, r;
W(Q--) {
l = read(), r = read();
printf("%d\n", query(T[l - ], T[r], , n, ));
}
}
return ;
}

        

主席树的妙用——Just h-index的更多相关文章

  1. SPOJ 3267 D-query(离散化+主席树求区间内不同数的个数)

    DQUERY - D-query #sorting #tree English Vietnamese Given a sequence of n numbers a1, a2, ..., an and ...

  2. bzoj1926[Sdoi2010]粟粟的书架 二分 主席树

    1926: [Sdoi2010]粟粟的书架 Time Limit: 30 Sec  Memory Limit: 552 MBSubmit: 1064  Solved: 421[Submit][Stat ...

  3. 主席树——求区间[l,r]不同数字个数的模板(向左密集 D-query)

    主席树的另一种用途,,(还有一种是求区间第k大,区间<=k的个数) 事实上:每个版本的主席树维护了每个值最后出现的位置 这种主席树不是以权值线段树为基础,而是以普通的线段树为下标的 /* 无修改 ...

  4. Codeforces 1000F One Occurrence 主席树|| 离线+线段树

    One Occurrence 为什么我半年前这么菜呀, 这种场只A三题... 我们在主席树 || 线段树上维护每个数的右边和它一样的数在哪里, 然后就变成了区间求最大值. 注意加进去的时候要把它右边一 ...

  5. ZOJ 2112 Dynamic Rankings (动态第k大,树状数组套主席树)

    Dynamic Rankings Time Limit: 10 Seconds      Memory Limit: 32768 KB The Company Dynamic Rankings has ...

  6. SPOJ 10628. Count on a tree (树上第k大,LCA+主席树)

    10628. Count on a tree Problem code: COT You are given a tree with N nodes.The tree nodes are number ...

  7. SPOJ 3267. D-query (主席树,查询区间有多少个不相同的数)

    3267. D-query Problem code: DQUERY English Vietnamese Given a sequence of n numbers a1, a2, ..., an  ...

  8. BZOJ3295: [Cqoi2011]动态逆序对(树状数组套主席树)

    3295: [Cqoi2011]动态逆序对 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 7465  Solved: 2662[Submit][Sta ...

  9. SPOJ - TTM 主席树

    给你一个系列\(a[1...n]\),要求可以区间求和,区间更新,也可以回溯过去 经典的主席树板子题,很久以前做的题了,代码太丑回炉重写 PS.题目标题To The Moon也是我最喜欢的游戏之一 这 ...

随机推荐

  1. java虚拟机开篇01

    一直以来对java 基础设施都啥都不知道啊,感觉有时候挺费力,挺吃劲的. 一下是一些很好的参考资料: http://blog.csdn.net/bingduanlbd/article/details/ ...

  2. 7.11 如何应用Varnish

    动态数据缓存 Step 1 修改devault.vcl文件 # This ) # man page for details on VCL syntax and semantics. # # Defau ...

  3. Java 将数据写入磁盘并读取磁盘上的文件

    package test; import java.io.BufferedReader;import java.io.FileReader;import java.io.FileWriter;impo ...

  4. C语言备忘录——取余与取模

    前几天,一个小姐姐问我取余和取模有什么区别,我当时第一反应就是二者是一样的,但是小姐姐咬死说不一样.我去百度了一下还真的不一样.脑壳疼,我当初误导了多少人.所以为了帮助我记忆也为了帮助预防我误人子弟 ...

  5. HYSBZ - 1588 营业额统计 (伸展树)

    题意:营业额统计 Tiger最近被公司升任为营业部经理,他上任后接受公司交给的第一项任务便是统计并分析公司成立以来的营业情况. Tiger拿出了公司的账本,账本上记录了公司成立以来每天的营业额.分析营 ...

  6. OpenMandriva或将放弃32位的支持

    导读 除了Ubuntu计划在其19.10发行版中删除32位包之外,OpenMandriva开发团队也是另一个备受关注的Linux发行版起草计划,以取消对32位包的支持. OpenMandriva计划在 ...

  7. vCenter组件和服务

    1).随VMware Platform Services Controller一起安装的服务 a. vCenter Single Sign-On身份验证服务 b. vSphere 许可证服务 c. V ...

  8. jQuery 里的 Promise

    1  $.ajax("test.html").done(function(){ alert("哈哈,成功了!"); }).fail(function(){ al ...

  9. yeoman 介绍、安装 和 使用

    一.介绍.安装 1, 是什么 Yeoman其实是3个工具的总和: ü  yo --- 脚手架,自动生成工具 ü  Grunt.gulp --- 构建工具 (最初只有grunt,后面gulp火了添加进来 ...

  10. BZOJ 3332

    题解:给边赋上权值,然后求最大生成树,如果不符合那就无解 证明:留坑 #include<iostream> #include<cstdio> #include<cstri ...