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

\(N,M\le50000\)

\(1\) 操作中 \(|c|\le N\)

\(2\) 操作中 \(|c|\le \text{maxlongint}\)

Solution

首先,操作一是给 \([a,b]\) 的每个位置插入一个数 \(c\),每个位置可以有多个数。(我一直以为是区间加QAQ。)

Solve(l, r, L, R) 表示第 \([L,R]\) 个询问可能的答案在区间 \([l,r]\) 中。每次把 \(c\le mid\) 的插入操作放到左边,把 \(c>mid\) 的放到右边;把答案在 \([l,mid]\) 的询问放到左边,把答案在 \([mid+1,r]\) 的询问放到右边。时间复杂度 \(O(n\log^2n)\)。

Code

#include <cstdio>

const int N = 50005;
typedef long long LL;
struct Node { int l, r, x, f; LL k; } a[N], b[N], c[N];
int ans[N], n, m, cnt; LL tag[N << 3], sum[N << 3]; int read() {
int x = 0, f = 1; char c = getchar();
while (c < '0' || c > '9') { if (c == '-') f = -1; c = getchar(); }
while (c >= '0' && c <= '9') x = (x << 3) + (x << 1) + (c ^ 48), c = getchar();
return x * f;
}
void pushdown(int cur, int l, int r) {
int mid = (l + r) >> 1;
tag[cur << 1] += tag[cur], tag[cur << 1 | 1] += tag[cur];
sum[cur << 1] += tag[cur] * (mid - l + 1), sum[cur << 1 | 1] += tag[cur] * (r - mid);
tag[cur] = 0;
}
void update(int cur, int l, int r, int L, int R, int x) {
if (L <= l && r <= R) { tag[cur] += x, sum[cur] += x * (r - l + 1); return; }
if (tag[cur]) pushdown(cur, l, r);
int mid = (l + r) >> 1;
if (L <= mid) update(cur << 1, l, mid, L, R, x);
if (mid < R) update(cur << 1 | 1, mid + 1, r, L, R, x);
sum[cur] = sum[cur << 1] + sum[cur << 1 | 1];
}
LL query(int cur, int l, int r, int L, int R) {
if (L <= l && r <= R) return sum[cur];
if (tag[cur]) pushdown(cur, l, r);
int mid = (l + r) >> 1; LL res = 0;
if (L <= mid) res = query(cur << 1, l, mid, L, R);
if (mid < R) res += query(cur << 1 | 1, mid + 1, r, L, R);
return res;
}
void solve(int l, int r, int L, int R) {
if (l > r || L > R) return;
if (l == r) {
for (int i = L; i <= R; ++i) if (a[i].f) ans[a[i].x] = n - l + 1;
return;
}
int mid = (l + r) >> 1, p = 0, q = 0;
for (int i = L; i <= R; ++i)
if (a[i].f) {
LL tmp = query(1, 1, n, a[i].l, a[i].r);
if (tmp >= a[i].k) b[++p] = a[i];
else a[i].k -= tmp, c[++q] = a[i];
} else {
if (a[i].k <= mid) update(1, 1, n, a[i].l, a[i].r, 1), b[++p] = a[i];
else c[++q] = a[i];
}
for (int i = 1; i <= p; ++i) if (!b[i].f) update(1, 1, n, b[i].l, b[i].r, -1);
for (int i = 1; i <= p; ++i) a[L + i - 1] = b[i];
for (int i = 1; i <= q; ++i) a[L + p + i - 1] = c[i];
solve(l, mid, L, L + p - 1), solve(mid + 1, r, L + p, R);
}
int main() {
n = read(), m = read();
for (int i = 1; i <= m; ++i) {
int op = read(), x = read(), y = read(); LL z; scanf("%lld", &z);
if (op == 1) a[i] = (Node){x, y, i, 0, n - z + 1};
else a[i] = (Node){x, y, ++cnt, 1, z};
}
solve(1, n, 1, m);
for (int i = 1; i <= cnt; ++i) printf("%d\n", ans[i]);
return 0;
}

[BZOJ 3110] [ZJOI 2013] K大数查询的更多相关文章

  1. BZOJ 3110 ZJOI 2013 K大数查询 树套树(权值线段树套区间线段树)

    题目大意:有一些位置.这些位置上能够放若干个数字. 如今有两种操作. 1.在区间l到r上加入一个数字x 2.求出l到r上的第k大的数字是什么 思路:这样的题一看就是树套树,关键是怎么套,怎么写.(话说 ...

  2. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

  3. 数据结构(树套树):ZJOI 2013 K大数查询

    有几个点卡常数…… 发现若第一维为位置,第二维为大小,那么修改时第一维修改区间,查询时第一维查询区间,必须挂标记.而这种情况下标记很抽象,而且Push_down不是O(1)的,并不可行. 那要怎么做呢 ...

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

    http://www.lydsy.com/JudgeOnline/problem.php?id=3110 题意:-- 思路:其实和之前POJ那道题差不多,只不过是换成区间更新,而且是第k大不是第k小, ...

  5. 解题:ZJOI 2013 K大数查询

    题面 树套树,权值线段树套序列线段树,每次在在权值线段树上的每棵子树上做区间加,查询的时候左右子树二分 本来想两个都动态开点的,这样能体现树套树在线的优越性.但是常数太大惹,所以外层直接固定建树了QA ...

  6. [ZJOI 2013] K大数查询

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=3110 [算法] 整体二分 + 线段树 时间复杂度 : O(NlogN ^ 2) [代 ...

  7. 【BZOJ 3110】 [Zjoi2013]K大数查询(整体二分)

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

  8. 【bzoj 3110】[Zjoi2013]K大数查询

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

  9. 【34.14%】【BZOJ 3110】 [Zjoi2013]K大数查询

    Time Limit: 20 Sec Memory Limit: 512 MB Submit: 5375 Solved: 1835 [Submit][Status][Discuss] Descript ...

随机推荐

  1. 阿里云 API调用实践(python语言)

    1.结论:阿里云的SDK开发,其实就是远程调用API,python的代码就是一个外壳,核心是封装成一个http报文,利用json格式,进行RPC调用. 2.SDK调用API的套路如下: # -*- c ...

  2. centos服务器如何监控访问ip,并将非法ip通过防火墙禁用

    centos服务器如何监控访问ip,并将非法ip通过防火墙禁用 上周给朋友帮忙,上架了一款小游戏(年年有鱼),项目刚一上线,就遇到了ddos攻击,阿里云连续给出了6次ddos预警提示,服务器一度处于黑 ...

  3. Java 环境部署之Idea :解决Idea 中gradle编译 war 包编码不对

    IDE进行Gradle操作,那么还需要设置IDE的参数.例如在IDEA中,需要打开File->Other Settings->Default Settings->Gradle,在Gr ...

  4. c/c++ socket API 调用后的错误判断 perror errno

    socket API 调用后的错误判断 perror errno 调用完socket API后,需要判断调用是否成功与失败.如果失败,会自动设置errno(是个整数), 并且用perror可以打印出具 ...

  5. Kafka 特性

    Kafka 特性 标签(空格分隔): Kafka 支持多个生产者 多个生成者连接Kafka来推送消息,这个和其他的消息队列功能基本上是一样的 支持多个消费者 Kafka支持多个消费者来读取同一个消息流 ...

  6. LeetCode算法题-Convert BST to Greater Tree(Java实现)

    这是悦乐书的第255次更新,第268篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第122题(顺位题号是538).给定二进制搜索树(BST),将其转换为更大树,使原始BS ...

  7. LeetCode算法题-Heaters(Java实现)

    这是悦乐书的第239次更新,第252篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第106题(顺位题号是475).冬天来了!您在比赛期间的第一份工作是设计一个固定温暖半径 ...

  8. Spark中的一些概念

    最近工作用到Spark,这里记一些自己接触到的Spark基本概念和知识. 本文链接:https://www.cnblogs.com/hhelibeb/p/10288915.html 名词 RDD:在高 ...

  9. HttpServletResponse简单理解

    Web服务器收到一个http请求,会针对每个请求创建一个HttpServletRequest和HttpServletResponse对象,从客户端取数据用HttpServletRequest,向客户端 ...

  10. [LeetCode] 8. 字符串转换整数 (atoi)

    题目链接:https://leetcode-cn.com/problems/string-to-integer-atoi/ 题目描述: 请你来实现一个 atoi 函数,使其能将字符串转换成整数. 首先 ...