【APIO2019】桥梁(询问分块)
Description
给定一张 \(n\) 个点,\(m\) 条边的无向图,边 \(i\) 的权值为 \(d_i\)。现有 \(q\) 次操作,第 \(j\) 个操作有两种模式:
- \(1\ b_j\ r_j\):将第 \(b_j\) 条边的权改为 \(r_j\)。
- \(2\ s_j\ w_j\):询问与点 \(s_j\) 连通的点的数量(包括自己)。若边 \((u, v)\) 的权值 \(d \ge w_j\),那么视作 \(u, v\) 连通。若 \((a, b), (b, c)\) 分别连通,那么 \((a, c)\) 也连通。
对于每个 \(2\) 操作输出答案。
Hint
- \(1\le n\le 5\times 10^4\)
- \(0\le m\le 10^5\)
- \(1\le q\le 10^5\)
- \(d, r, w\in [1, 10^9]\)
Solution
先看一个比较显然的部分分做法:将询问离线,按 \(w\) 排序,边按 \(d\) 排序,降序。如果没有询问那这个就是 naive,而正好有这档部分分。考虑修改:将边按是否有修改分类,无修直接加,带修在处理询问时暴力枚举,根据操作的时间做修改,然后一个个加入并查集,最后撤回。由于需要可撤销,需要启发式合并。复杂度 \(O(q\log q + m\log m + q^2\log n)\)。数据分治期望有 \(27\) pts。
一次性枚举所有询问,这就是复杂度瓶颈所在。但如果直接暴力就得扫一遍所有边。
基于上述解法,考虑根号平衡的思想来优化——对询问分块!
设块长为 \(s\)。对于当前块,我们默认前面块已经处理好修改操作了,那么就不需要再考虑前面的块的修改。首先任然是两次排序,一次所有边,一次 块内的询问,那么复杂度是 \(O(m\log m + s\log s)\)。枚举询问时,对于所有无修边,直接加入;对于带修边,由分了块,带修边不会太多,我们直接枚举,把有效的边插入,统计答案后再撤销即可,这里复杂度为 \(O(s^2\log s)\)。
当前块处理完后,清空询问与修改。这样的算法总复杂度为 \(O(s(s^2\log s + s\log s + m\log m)) = O(s^3\log s + sm\log m)\)。
当 \(s = \sqrt{q}\) 是,复杂度为 \(O(q\sqrt{q}\log q + m\sqrt{q} \log m)\)。
实现细节较多,需要注意。
Code
/*
* Author : _Wallace_
* Source : https://www.cnblogs.com/-Wallace-/
* Problem : APIO2019 桥梁
*/
#include <algorithm>
#include <cmath>
#include <iostream>
#include <vector>
using namespace std;
const int N = 5e4 + 5;
const int M = 1e5 + 5;
int n, m, q;
struct Edge {
int u, v, w, idx;
} e[M];
struct Command {
int pos, val, idx;
}; vector<Command> Q, E;
int ans[M];
int fa[N], siz[N], stk[N], top = 0;
int find(int x) {
return x == fa[x] ? x : find(fa[x]);
}
void merge(int x, int y) {
x = find(x), y = find(y);
if (x == y) return;
if (siz[x] < siz[y]) swap(x, y);
stk[++top] = y;
siz[x] += siz[y], fa[y] = x;
}
void backtrack(int time) {
while (top > time) {
int x = stk[top--];
siz[fa[x]] -= siz[x];
fa[x] = x;
}
}
void reset() {
for (int i = 1; i <= n; i++)
fa[i] = i, siz[i] = 1;
top = 0;
}
int cur[M];
int ref[M];
void solve() {
sort(e + 1, e + 1 + m, [](const Edge& a, const Edge& b) {
return a.w > b.w;
});
sort(Q.begin(), Q.end(), [](const Command& a, const Command& b) {
return a.val > b.val;
});
for (int i = 1; i <= m; i++)
ref[e[i].idx] = i;
reset();
vector<Command> E_tmp;
for (auto t : E) {
cur[t.pos] = -1;
E_tmp.push_back(Command{t.pos, e[ref[t.pos]].w, 0});
}
for (auto t : E) E_tmp.push_back(t);
for (int i = 0, j = 1; i < int(Q.size()); i++) {
for (; j <= m && e[j].w >= Q[i].val; j++)
if (!cur[e[j].idx]) merge(e[j].u, e[j].v);
for (auto t : E_tmp) if (t.idx <= Q[i].idx)
cur[t.pos] = t.val;
int bt = top; // backtrack
for (auto t : E) if (cur[t.pos] >= Q[i].val)
merge(e[ref[t.pos]].u, e[ref[t.pos]].v);
ans[Q[i].idx] = siz[find(Q[i].pos)];
backtrack(bt);
}
for (auto t : E) e[ref[t.pos]].w = t.val, cur[t.pos] = 0;
E.clear(), Q.clear();
}
signed main() {
ios::sync_with_stdio(false);
cin >> n >> m;
for (int i = 1; i <= m; i++) {
cin >> e[i].u >> e[i].v >> e[i].w;
e[i].idx = i;
}
cin >> q;
int gap = int(sqrt(q)) << 2;
for (int i = 1; i <= q; i++) {
int op, pos, val;
cin >> op >> pos >> val;
if (op == 1) E.push_back(Command{pos, val, i});
else Q.push_back(Command{pos, val, i});
if (i % gap == 0) solve();
}
if (q % gap != 0) solve();
for (int i = 1; i <= q; i++)
if (ans[i]) printf("%d\n", ans[i]);
return 0;
}
【APIO2019】桥梁(询问分块)的更多相关文章
- 【LOJ#3145】[APIO2019]桥梁(分块,并查集)
[LOJ#3145][APIO2019]桥梁(分块,并查集) 题面 LOJ 题解 因为某个\(\text{subtask}\)没判\(n=1\)的情况导致我自闭了很久的题目... 如果没有修改操作,可 ...
- BZOJ3720 Gty的妹子树 询问分块、主席树
传送门 学到了询问分块的科技-- 对于修改操作,每发生了\(S\)次修改就重构整棵树,小于\(S\)次的修改操作丢到一个队列里面. 对于每一次查询操作,先在主席树上查询当前子树内部大于\(k\)的节点 ...
- CodeForces 103D Time to Raid Cowavans 询问分块
Time to Raid Cowavans 题意: 询问 下标满足 a + b * k 的和是多少. 题解: 将询问分块. 将b >= blo直接算出答案. 否则存下来. 存下来之后,对于每个b ...
- P5443 [APIO2019]桥梁 [分块+并查集]
分块+并查集,大板子,没了. 并查集不路径压缩,可撤销,然后暴力删除 这样对于每个块都是独立的,所以直接搞就行了. 然后块内修改操作搞掉,就是单独的了 // powered by c++11 // b ...
- bzoj 3781: 小B的询问 分块
3781: 小B的询问 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 196 Solved: 135[Submit][Status] Descrip ...
- 题解-APIO2019桥梁
problem \(\mathrm {loj-3145}\) 题意概要:给定一张 \(n\) 点 \(m\) 边的无向图,边有边权,共 \(q\) 次操作,每次会将第 \(x\) 条边的权值改为 \( ...
- P5443 [APIO2019]桥梁
传送门 子任务 $4$ 告诉我们可以离线搞带权并查集 从大到小枚举询问,从大到小连边 如果没有修改操作就可以过了 但是有修改,考虑最暴力的暴力,搞可撤销并查集 同样先离线,从大到小处理询问时,按原边权 ...
- 题解 洛谷 P5443 【[APIO2019]桥梁】
考虑若只有查询操作,那么就可以构造\(Kruskal\)重构树,然后在线询问了,也可以更简单的把询问离线,把询问和边都按权值从大到小排序,然后双指针依次加入对于当前询问合法的边,用并查集维护每个点的答 ...
- 【对询问分块】【主席树】bzoj2683 简单题
对操作序列分块,每S次暴力重建主席树. 当S=sqrt(n*log(n))时,复杂度为O(m*sqrt(n*log(n))). 在线的. #include<cstdio> #include ...
随机推荐
- Linux(centos6.8)配置Mysql环境
1.下载mysql安装包 https://downloads.mysql.com/archives/community/ 2.查询mysql信息 [1]查询mysql是否已经安装 [root@plut ...
- HDU100题简要题解(2020~2029)
HDU2020 绝对值排序 题目链接 Problem Description 输入n(n<=100)个整数,按照绝对值从大到小排序后输出.题目保证对于每一个测试实例,所有的数的绝对值都不相等. ...
- centos8 连接wifi
从官网下载的6G多的iso安装后 #ifconfig -a 如下 enp3s0: flags=4163<UP,BROADCAST,RUNNING,MULTICAST> mtu 1500l ...
- 还不懂Java高并发的,建议看看这篇阿里大佬的总结,写的非常详细
前言 进程是计算机中程序关于某几何数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位.是操作系统结构的基础 线程可以说是轻量级的进程,是程序执行的最小单位,使用多线程而不用多进程去进行并发程 ...
- 【PYTHON】操作excel笔记
import xlrd book = xlrd.open_workbook('..//data//case1.xls') # 获取excel对象 print(book.sheets()) # 获取ex ...
- MySQL给临时表分组后Max函数无效
有道练习题"取得平均薪水最高的部门的部门编号(至少给出两种解决方案)", 为什么我给临时表分组后Max函数就无效了?不分组就可以,但是无法查询到DEPTNO,MySQL版本8.0+ ...
- php8.0正式版新特性和性能优化学习
前言 PHP团队宣布PHP8正式GA(链接).php的发展又开启了新的篇章,PHP8.0.0版本引入了一些重大变更及许多新特性和性能优化机制.火速学习下~ JIT(Just in Time Compi ...
- Luogu P2656 采蘑菇
尽管是缩点的习题,思路也是在看了题解后才明白的. 首先,每个强连通分量内的点都是一定互通的,也就是可以完全把这里面的边都跑满,摘掉所有能摘的蘑菇.那么,考虑给每一个强连通分量化为的新点一个点权,代表摘 ...
- 2020.11.26 IntellJ idea激活码失效解决方法(最新idea激活码及安装参数!)
今天是2020年11月26号,小伙伴们是不是有发现自己的idea激活码失效了,不瞒大家,小编也是一个JAVA开发者,到了公司打开idea,然后就发现事情不妙,经过1个多小时的摸索,终于把最近的安装参数 ...
- 企业安全06-Apache Log4j Server 反序列化命令执行漏洞(CVE-2017-5645)
CVE-2017-5645 Apache Log4j Server 反序列化命令执行漏洞(CVE-2017-5645) 一.漏洞原理 Apache Log4j是一个用于Java的日志记录库,其支持启动 ...