日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪(fǎng)問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

LCA题集

發(fā)布時(shí)間:2025/7/14 编程问答 20 豆豆
生活随笔 收集整理的這篇文章主要介紹了 LCA题集 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

點(diǎn)的距離(模板題)

樹(shù)中兩點(diǎn)間的距離就是d[u] + d[v] - 2 * d[lca(u, v)]

#include<bits/stdc++.h> #define REP(i, a, b) for(register int i = (a); i < (b); i++) #define _for(i, a, b) for(register int i = (a); i <= (b); i++) using namespace std;const int MAXN = 1e5 + 10; const int MAXM = 20; struct Edge { int to, next; }; Edge e[MAXN << 1]; int head[MAXN], num, n; int up[MAXN][MAXM + 10], d[MAXN];void AddEdge(int from, int to) {e[num] = Edge{to, head[from]};head[from] = num++; }void dfs(int u, int fa) {for(int i = head[u]; ~i; i = e[i].next){int v = e[i].to;if(v == fa) continue;d[v] = d[u] + 1;up[v][0] = u;dfs(v, u);} }void get_up() {_for(j, 1, MAXM)_for(i, 1, n)up[i][j] = up[up[i][j-1]][j-1]; }int lca(int u, int v) {if(d[u] < d[v]) swap(u, v);for(int i = MAXM; i >= 0; i--)if(d[up[u][i]] >= d[v]) u = up[u][i];if(u == v) return u;for(int i = MAXM; i >= 0; i--)if(up[u][i] != up[v][i]) u = up[u][i], v = up[v][i];return up[u][0]; }int main() {memset(head, -1, sizeof(head)); num = 0; scanf("%d", &n);REP(i, 1, n){int u, v;scanf("%d%d", &u, &v);AddEdge(u, v); AddEdge(v, u); }dfs(1, -1);get_up();int q; scanf("%d", &q);while(q--){int u, v;scanf("%d%d", &u, &v);printf("%d\n", d[u] + d[v] - 2 * d[lca(u, v)]);}return 0; }

暗的連鎖

這道題首先有個(gè)轉(zhuǎn)化

切兩刀能不能切斷,取決于非樹(shù)邊,因?yàn)榉菢?shù)邊會(huì)構(gòu)成環(huán)

那么可以把非樹(shù)邊構(gòu)成的環(huán)上所有的樹(shù)邊都覆蓋一次

如果只覆蓋一次,那么顯然有唯一解

如果沒(méi)有被覆蓋,那就加上非樹(shù)邊的數(shù)目,因?yàn)榈诙犊梢郧腥我庖粭l非樹(shù)邊

如果覆蓋兩次以上,那么兩刀是不能解決問(wèn)題的。

所以就有維護(hù)每條邊被覆蓋了多少次

這里用到了樹(shù)上差分,f[u]表示u與u的父親連的邊的覆蓋次數(shù)

對(duì)于非樹(shù)邊u, v,讓f[u]++, f[v]++, f[lca(u, v)]--

最后在“前綴和回來(lái)”,即統(tǒng)計(jì)所有子樹(shù)的值加起來(lái),就是答案

最后注意計(jì)算答案的時(shí)候根的f數(shù)組不算,因?yàn)楦鶝](méi)有父親

#include<bits/stdc++.h> #define REP(i, a, b) for(register int i = (a); i < (b); i++) #define _for(i, a, b) for(register int i = (a); i <= (b); i++) using namespace std;const int MAXN = 1e5 + 10; const int MAXM = 20; struct Edge{ int to, next; }; Edge e[MAXN << 1]; int head[MAXN], num, n, m; int up[MAXN][MAXM + 10], d[MAXN]; int f[MAXN];void read(int& x) {int f = 1; x = 0; char ch = getchar();while(!isdigit(ch)) { if(ch == '-') f = -1; ch = getchar(); }while(isdigit(ch)) { x = x * 10 + ch - '0'; ch = getchar(); }x *= f; }void AddEdge(int to, int from) {e[num] = Edge{to, head[from]};head[from] = num++; }void dfs(int u, int fa) {for(int i = head[u]; ~i; i = e[i].next){int v = e[i].to;if(v == fa) continue;d[v] = d[u] + 1;up[v][0] = u;dfs(v, u);} }int dp(int u, int fa) {for(int i = head[u]; ~i; i = e[i].next){int v = e[i].to;if(v == fa) continue;dp(v, u);f[u] += f[v];} }void init() {dfs(1, -1);_for(j, 1, MAXM)_for(i, 1, n)up[i][j] = up[up[i][j - 1]][j - 1]; }int lca(int u, int v) {if(d[u] < d[v]) swap(u, v);for(int i = MAXM; i >= 0; i--)if(d[up[u][i]] >= d[v])u = up[u][i];if(u == v) return u;for(int i = MAXM; i >= 0; i--)if(up[u][i] != up[v][i])u = up[u][i], v = up[v][i];return up[u][0]; }int main() {read(n);read(m);memset(head, -1, sizeof(head)); num = 0;REP(i, 1, n){int u, v; read(u), read(v);AddEdge(u, v); AddEdge(v, u); }init(); _for(i, 1, m){int u, v; read(u), read(v);f[u]++; f[v]++; f[lca(u, v)] -= 2;}dp(1, -1);int ans = 0;_for(i, 2, n){if(f[i] == 1) ans++;if(f[i] == 0) ans += m;}printf("%d\n", ans);return 0; }

?

轉(zhuǎn)載于:https://www.cnblogs.com/sugewud/p/9819317.html

總結(jié)

以上是生活随笔為你收集整理的LCA题集的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。