Solution -「NOI.AC 省选膜你赛」union
题目
题意简述
给定两颗树 \(A,B\),\(A\) 中的任一结点 \(u\) 与 \(B\) 中的任一结点 \(v\) 都有一个关系值 \(f(u,v)\),初始为 \(0\)。再给出 \(q\) 个形如 \(a1,b1,a2,b2,c\) 的操作,表示对于 \(A\) 中路径 \(a1\leftrightarrow b1\) 上的任一结点 \(u\) 和 \(B\) 中路径 \(a2\leftrightarrow b2\) 上的任一结点 \(v\),\(f(u,v)\leftarrow f(u,v)+c\)。求操作完成后所有的 \(f(u,v)\)。
数据规模
设 \(A\) 的结点数 \(n\),\(B\) 的结点数 \(m\)。
\(n,m\le10^4;~q\le5\times10^5\)。
题解
一道有意思的差分题 owo。
首先将 \(A\) 和 \(B\) 分别树链剖分,把结点编号转化为 \(DFN\) 来考虑问题。
形象地,我们把 \(f\) 列成一个表格,第 \(i\) 行第 \(j\) 列的值表示 \(A\) 中 \(DFN\) 为 \(i\) 的结点 \(u\) 和 \(B\) 中 \(DFN\) 为 \(j\) 的结点 \(v\) 的 \(f(u,v)\) 值。如果操作涉及路径的 \(DFN\) 值连续,那么就相当于修改这个表格的一个子矩阵,可以差分做到。推广到一般情况,只需要用树剖取出 \(A\) 中路径所覆盖的若干个 \(DFN\) 区间和 \(B\) 中路径覆盖的若干个 \(DFN\) 区间,暴力地两两配对,修改子矩阵即可。最后前缀和还原表格,就求出每一个 \(f(u,v)\) 啦~
复杂度 \(O(q\log^2n)\)。
代码
#include <cstdio>
#include <vector>
#include <utility>
#define x1 tmpx1
#define y1 tmpy1
#define x2 tmpx2
#define y2 tmpy2
inline int rint () {
int x = 0, f = 1; char s = getchar ();
for ( ; s < '0' || '9' < s; s = getchar () ) f = s == '-' ? -f : f;
for ( ; '0' <= s && s <= '9'; s = getchar () ) x = x * 10 + ( s ^ '0' );
return x * f;
}
template<typename Tp>
inline void wint ( Tp x ) {
if ( x < 0 ) putchar ( '-' ), x = ~ x + 1;
if ( 9 < x ) wint ( x / 10 );
putchar ( x % 10 ^ '0' );
}
const int MAXN = 1e4;
struct TreeSplit {
int n, fa[MAXN + 5], dep[MAXN + 5], son[MAXN + 5], siz[MAXN + 5];
int indx, top[MAXN + 5], dfn[MAXN + 5], ref[MAXN + 5];
std :: vector<int> graph[MAXN + 5];
inline void read () {
n = rint ();
for ( int i = 1, u, v; i < n; ++ i ) {
u = rint (), v = rint ();
graph[u].push_back ( v ), graph[v].push_back ( u );
}
}
inline void DFS1 ( const int u, const int f ) {
dep[u] = dep[fa[u] = f] + ( siz[u] = 1 );
for ( int v: graph[u] ) if ( v ^ f ) {
DFS1 ( v, u ), siz[u] += siz[v];
if ( siz[son[u]] < siz[v] ) son[u] = v;
}
}
inline void DFS2 ( const int u, const int tp ) {
top[ref[dfn[u] = ++ indx] = u] = tp;
if ( son[u] ) DFS2 ( son[u], tp );
for ( int v: graph[u] ) if ( v ^ fa[u] && v ^ son[u] ) DFS2 ( v, v );
}
inline std :: vector<std :: pair<int, int> > getPath ( int u, int v ) {
static std :: vector<std :: pair<int, int> > ret; ret.clear ();
while ( top[u] ^ top[v] ) {
if ( dep[top[u]] < dep[top[v]] ) u ^= v ^= u ^= v;
ret.push_back ( { dfn[top[u]], dfn[u] } ), u = fa[top[u]];
}
if ( dep[u] < dep[v] ) u ^= v ^= u ^= v;
return ret.push_back ( { dfn[v], dfn[u] } ), ret;
}
} T1, T2;
int sum[MAXN + 5][MAXN + 5];
inline void add ( const int x1, const int y1, const int x2, const int y2, const int k ) {
sum[x1][y1] += k, sum[x1][y2 + 1] -= k, sum[x2 + 1][y1] -= k, sum[x2 + 1][y2 + 1] += k;
}
int main () {
T1.read (), T2.read ();
T1.DFS1 ( 1, 0 ), T1.DFS2 ( 1, 1 );
T2.DFS1 ( 1, 0 ), T2.DFS2 ( 1, 1 );
std :: vector<std :: pair<int, int> > pathA, pathB;
for ( int q = rint (), a, b, u, v, k; q --; ) {
a = rint (), b = rint (), u = rint (), v = rint (), k = rint ();
pathA = T1.getPath ( a, b ), pathB = T2.getPath ( u, v );
for ( auto seca: pathA ) for ( auto secb: pathB ) {
add ( seca.first, secb.first, seca.second, secb.second, k );
}
}
for ( int i = 1; i <= T1.n; ++ i ) {
for ( int j = 1; j <= T2.n; ++ j ) {
sum[i][j] += sum[i - 1][j] + sum[i][j - 1] - sum[i - 1][j - 1];
}
}
long long ans = 0;
for ( int i = 1; i <= T1.n; ++ i ) {
for ( int j = 1; j <= T2.n; ++ j ) {
ans ^= 1ll * T1.ref[i] * T2.ref[j] * sum[i][j];
}
}
wint ( ans ), putchar ( '\n' );
return 0;
}
Solution -「NOI.AC 省选膜你赛」union的更多相关文章
- Solution -「NOI.AC 省选膜你赛」T2
这道题就叫 T2 我有什么办法www 题目 题意简述 给定一个字符串 \(s\),其长度为 \(n\),求无序子串对 \((u,v)\) 的个数,其中 \((u,v)\) 满足 \(u,v\) ...
- Solution -「NOI.AC 省选膜你赛」array
题目 题意简述 维护一个长度为 \(n\) 的序列 \(\{a_n\}\),并给出 \(q\) 个操作: 将下标为 \(x\) 的数修改为 \(y\). 给定 \(l,r,k\),求最大的 \(m ...
- Solution -「NOI.AC 省选膜你赛」寄蒜几盒
题目 题意简述 给定一个含有 \(n\) 个顶点的凸多边形( \(n\) 是偶数),对于每一对相对的边(即中间有 \(\frac{n}2-1\) 条其它边),延长它们以将平面分割为多块,并把包含原 ...
- cdcqの省选膜你赛
cdcqの省选膜你赛 比赛当天因为在杠hnoi2016的大数据结构没有参加,今天补了一下.挺好玩的虽然不看一句话题意的话真的卡读题 此生无悔入东方,来世愿生幻想乡 2651. 新史「新幻想史 -现代史 ...
- Solution -「NOI 2021」「洛谷 P7740」机器人游戏
\(\mathcal{Description}\) Link. 自己去读题面叭~ \(\mathcal{Solution}\) 首先,参悟[样例解释 #2].一种暴力的思路即为钦定集合 \ ...
- Solution -「NOI 2020」「洛谷 P6776」超现实树
\(\mathcal{Description}\) Link. 对于非空二叉树 \(T\),定义 \(\operatorname{grow}(T)\) 为所有能通过若干次"替换 \( ...
- Solution -「NOI 模拟赛」彩色挂饰
\(\mathcal{Description}\) 给定一个含 \(n\) 个点 \(m\) 条边的简单无向图,设图中最大点双的大小为 \(s\),则保证 \(s\le6\).你将要用 \(k\) ...
- Solution -「NOI 模拟赛」出题人
\(\mathcal{Description}\) 给定 \(\{a_n\}\),求一个 \(\{b_{n-1}\}\),使得 \(\forall x\in\{a_n\},\exists i,j\ ...
- Solution -「NOI 2016」「洛谷 P1587」循环之美
\(\mathcal{Description}\) Link. 给定 \(n,m,k\),求 \(x\in [1,n]\cap\mathbb N,y\in [1,m]\cap \mathbb ...
随机推荐
- 面试题68 - II. 二叉树的最近公共祖先
<搜索树结点> <获取路径> 题目描述 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先. 百度百科中最近公共祖先的定义为:"对于有根树 T 的两个结点 p.q ...
- SQL高级优化(五)之执行计划
一.explain 执行计划:在MySQL中可以通过explain关键字模拟优化器执行SQL语句,从而知道MySQL是如何处理SQL语句的. explain:MySQL执行计划的工具,查看MySQL如 ...
- 在实验中观察指针——C++ 函数参数的压栈顺序
前言 好久没写东西了,突发奇想,写写函数参数的压栈顺序 先看看这个问题 https://q.cnblogs.com/q/137133/ 然后看我简化的代码,猜输出结果是多少? #include< ...
- react中引入图片路劲正确但是页面上不显示或者打包后不能正常显示的问题
一.react中图片引入方式 以前我们用img引入图片只需要如下即可,在react中这样写会报错: <img src="../assets/zzsc1.png" /> ...
- 为什么JavaWeb要分层
首先bai让我们坐着时光机回到n年前的web开发.那个时候最早du都是静态的html页面,zhi后来有了数据库,有了所谓dao的动态页面,然后程序猿在编码的时候,会把所有的代码都写在页面上,包括数据库 ...
- 与Elasticsearch交互的客户端
1.访问ES的方式 访问es的方式有两种,一种是http方式,还有一种是java客户端方式. 其中Java客户端又分为:1.1.Node client: 节点客户端实际上是一个集群中的节点(但不保存数 ...
- vuecli学习01 - 环境搭建
到这个链接下载nvm的安装包:https://github.com/coreybutler/nvm-windows/releases. 然后点击一顿下一步,安装即可! 安装完成后,还需要配置环境变量. ...
- Understanding C++ Modules In C++20 (1)
Compiling evironment: linux (ubuntu 16.04)+ gcc-10.2. The Post will clarify and discuss what modules ...
- js源码-数组中的push()和unshift()方法的源码实现
人话不多,直接上代码,在代码中解析,不足之处请谅解: push() Array.prototype._push=function(...value){//在Array原型链上添加_push方法 for ...
- Maven仓库的目录结构
_remote.repositories文件 本地库中的包都有一个_remote.repositories文件,示例: #NOTE: This is an Aether internal implem ...