终于入门整体二分了,勉勉强强算是搞懂了一个题目吧。

整体二分很多时候可以比较好的离线处理区间\(K\)大值的相关问题。考虑算法流程:

操作队列\(arr\),其中有询问和修改两类操作。

每次在答案的可行值域上二分一个\(mid\),把询问的答案\(>mid\)的分在\(R\)部,\(<=mid\)的分在\(L\)部。把修改的值\(>mid\)的分在\(R\)部,\(<=mid\)的分在\(L\)部。

何谓整体二分?就是直接一起二分所有的询问操作的答案,然后暴力扫描当前操作区间,将其划分为答案的左子区间与右子区间两个部分。

那么以什么为划分依据呢?看看这个操作对于左子区间有没有贡献。如果没有,那么就划分到右子区间中,然后将这个操作的权值更改为这个贡献减去所需的贡献,反之,则划分到左子区间中,同时将这个操作的贡献加入某一个容器,为询问操作服务。

这么说可能有点晕。就这道题说的话,应该是这样:

我们设尚未解决的操作区间为\([ql,qr]\),答案区间为[l,r][l,r],令当前答案为\(mid\)。

则若该操作是添加操作,如果其添加的\(C<=mid\),这此次操作对于左子区间有贡献,加入左子区间中,并将区间线段树中的区间\([q[i].l,q[i].r]\)整体加\(1\).

反之,则将操作加入到右子区间中。

若该操作是询问操作,如果当前的\(mid\)在线段树中查询到的,比它大的数的个数\(query()>=q[i].k\),则证明该询问操作应该在右子区间内可以找到答案。反之,则将\(q[i].k-=query()\),减去此次查询的贡献,然后将询问操作添加到左子区间中。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std; #define int long long
const int N = 50000 + 5; struct Ask {
int l, r, v, id, op; void read () {
cin >> op >> l >> r >> v;
} }q[N], tl[N], tr[N]; int ans[N], tag[N << 2], clr[N << 2], sum[N << 2]; int n, m, Q; #define lc (p << 1)
#define rc (p << 1 | 1)
#define mid ((l + r) >> 1) void pushdown (int p, int l, int r) {
if (clr[p]) {
clr[p] = 0;
tag[lc] = tag[rc] = 0;
sum[lc] = sum[rc] = 0;
clr[lc] = 1, clr[rc] = 1;
}
if (tag[p]) {
tag[lc] += tag[p];
tag[rc] += tag[p];
sum[lc] += tag[p] * (mid - l + 1);
sum[rc] += tag[p] * (r - mid);
tag[p] = 0;
}
} void push_up (int p) {
sum[p] = sum[lc] + sum[rc];
} void modify (int ql, int qr, int w, int p = 1, int l = 1, int r = n) {
if (ql <= l && r <= qr){
tag[p] += w;
sum[p] += w * (r - l + 1);
return;
}
pushdown (p, l, r);
if (ql <= mid) modify (ql, qr, w, lc, l, mid);
if (mid < qr) modify (ql, qr, w, rc, mid + 1, r);
push_up (p);
} int query (int ql, int qr, int p = 1, int l = 1, int r = n) {
if (ql <= l && r <= qr) {
return sum[p];
}
int ret = 0; pushdown(p,l,r);
if (ql <= mid) ret += query (ql, qr, lc, l, mid);
if (mid < qr) ret += query (ql, qr, rc, mid + 1, r);
return ret;
} void solve (int st, int en, int l, int r) {
// [st, en] -> 处理操作的左右端点
// [l, r] -> 对应值域
if (l == r) {
for (int i = st; i <= en; ++i) {
if (q[i].op == 2) ans[q[i].id] = l; // 查询
}
return;
}
int L = 0, R = 0;
clr[1] = 1; tag[1] = sum[1] = 0;
for (int i = st; i <= en; ++i) {
if (q[i].op == 1){
if (q[i].v > mid) { // > mid 的操作对于答案 <= mid 的询问不会影响
modify (q[i].l, q[i].r, 1);
tr[++R] = q[i];
} else {
tl[++L] = q[i];
}
} else {
int val = query (q[i].l, q[i].r);
if (val < q[i].v){
q[i].v -= val;
tl[++L] = q[i]; // L部答案 <= mid
}else{
tr[++R] = q[i]; // R部答案 > mid
}
}
}
for (int i = 1; i <= L; ++i) {
q[st + i - 1] = tl[i];
}
for (int i = L + 1; i <= L + R; ++i) {
q[st + i - 1] = tr[i - L];
}
solve (st, st + L - 1, l, mid);
solve (st + L, en, mid + 1, r);
} signed main () {
cin >> n >> m;
for (int i = 1; i <= m; ++i) {
q[i].read ();
if (q[i].op == 2) {
q[i].id = ++Q;
}
}
solve (1, m, -n, n);
for (int i = 1; i <= Q; ++i) {
cout << ans[i] << endl;
}
}

P3332 [ZJOI2013]K大数查询 整体二分的更多相关文章

  1. BZOJ 3110: [Zjoi2013]K大数查询 [整体二分]

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

  2. BZOJ3110:[ZJOI2013]K大数查询(整体二分)

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

  3. BZOJ.3110.[ZJOI2013]K大数查询(整体二分 树状数组/线段树)

    题目链接 BZOJ 洛谷 整体二分求的是第K小(利用树状数组).求第K大可以转为求第\(n-K+1\)小,但是这样好像得求一个\(n\). 注意到所有数的绝对值\(\leq N\),将所有数的大小关系 ...

  4. [ZJOI2013]K大数查询——整体二分

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

  5. BZOJ 3110 [Zjoi2013]K大数查询 ——整体二分

    [题目分析] 整体二分显而易见. 自己YY了一下用树状数组区间修改,区间查询的操作. 又因为一个字母调了一下午. 貌似树状数组并不需要清空,可以用一个指针来维护,可以少一个log 懒得写了. [代码] ...

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

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

  7. BZOJ3110:[ZJOI2013]K大数查询(整体二分版)

    浅谈离线分治算法:https://www.cnblogs.com/AKMer/p/10415556.html 题目传送门:https://lydsy.com/JudgeOnline/problem.p ...

  8. BZOJ 3110 [ZJOI2013]K大数查询 (整体二分+线段树)

    和dynamic rankings这道题的思想一样 只不过是把树状数组换成线段树区间修改,求第$K$大的而不是第$K$小的 这道题还有负数,需要离散 #include <vector> # ...

  9. 洛谷 P3332 [ZJOI2013]K大数查询 解题报告

    P3332 [ZJOI2013]K大数查询 题目描述 有\(N\)个位置,\(M\)个操作.操作有两种,每次操作如果是\(\tt{1\ a\ b\ c}\)的形式表示在第\(a\)个位置到第\(b\) ...

随机推荐

  1. Interface default method介绍

    一.introduce interface default method Introduce default methodWrite the default method at interfaceTh ...

  2. ForkJoin使用

    一.Fork Join 分而治之的办法 JDk为Fork/Join框架提供了很好的支持,我们想要用这个算法首先得创建一个Fork/Join任务,在JDK中这个任务就叫做:ForJoinTask,只要继 ...

  3. Linux日志筛选命令

    (1)Linux目录操作命令 cd ..退出当前目录,返回上一级目录:cd / 退出当前目录,返回根目录: mkdir命令用于创建一个新的目录:rmdir命令功能删除指定的空目录. (2)Linux筛 ...

  4. Java学习之==>集合体系(待续。。)

    一.概述 Java的集合体系,本质上是一个陈放数据的容器,像之前学过的数组也是陈放数据的容器,但在 Java 中数组的长度是固定的,使用起来没那么方便.集合提供了更加强大的功能,使用起来也更方便和快捷 ...

  5. opencv的曲线拟合polyfit

    推荐一个不错的网页,可以直接用solve函数求解方程组: http://m.blog.csdn.net/u014652390/article/details/52789591 4.1 曲线拟合的最小二 ...

  6. 【VS开发】【图像处理】 bayer, yuv, RGB转换方法

    因为我的STVxxx USB camera输出格式是bayer格式,手头上只有YUVTOOLS这个查看工具,没法验证STVxxx在开发板上是否正常工作. 网上找了很久也没找到格式转换工具,最后放弃了, ...

  7. [转帖]黑客通过 Rootkit 恶意软件感染超 5 万台 MS-SQL 和 PHPMyAdmin 服务器

    黑客通过 Rootkit 恶意软件感染超 5 万台 MS-SQL 和 PHPMyAdmin 服务器 https://www.cnbeta.com/articles/tech/852141.htm 病毒 ...

  8. SPOJ 4003 Phone List 题解

    题面 啊~,很水的一道trie树模板题: 当两个串存在关系时情况有两种: 若当前串插入后没有任何新建节点,则该串肯定是之前插入的某个串的前缀: 若在插入的时候,有某个经过的节点带有某串结尾的标记,则之 ...

  9. MySQL总结(5)

    视图 SELECT cust_name,cust_contact FROM customers,orders,orderitems WHERE customers.cust_id=orders.cus ...

  10. docker安装笔记

    Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的镜像中,然后发布到任何流行的 Linux或Windows 机器上,也可以实现虚拟化.容器是完全使用沙箱机制,相 ...