原题链接:蓝桥杯2023年第十四届省赛真题-砍树
解题思路:
本题写一个链式前向星的写法,仅供参考
由于我们知道在一棵树上任意两个点的路径经过的边是唯一确定的因此对于每一个路径我们对路径上的边的边权加一,这里我们可以通过树上差分在O(n)复杂度内解决,一共m对因此如果说某个边被经过了m次则说明在这m条路径上每一条都需要经过这个边,又因为在树上路径的唯一性我们可以得知这个边是必须边删掉它会使得所有点对组成的路径断开。
不了解树上差分边差分算法的可以移步oiwiki网址 https://oi-wiki.org/basic/prefix-sum/
注意事项:
参考代码:
#include <bits/stdc++.h> using namespace std; #define all(c) (c).begin(), (c).end() #define rall(x) (x).rbegin(), (x).rend() #define sz(x) (int)(x).size() #define Sum(a) accumulate((a).begin(), (a).end() , 0ll) #define Min(a) *std::min_element((a).begin(), (a).end()) #define Max(a) *std::max_element((a).begin(), (a).end()) #define rev(a) reverse((a).begin(), (a).end()) #define each(x, a) for(auto& x : a) #define mst(a, x) memset(a, x, sizeof(a)) #define LL long long #define rep(i, from, to) for(ll i = from;i<to;i++) #define rrep(i, from, to) for(ll i = from;i>to;i--) #define to_uni(a) a.erase(unique(begin(a), end(a)), end(a)) #define fi first #define se second #define mp make_pair #define pb push_back #define pp pop_back #define endl "\n" typedef pair<int, int> PII; typedef pair<LL, LL> PLL; const int mod = 1e9 + 7; const int P = 998244353; const int INF = 0x3f3f3f3f; const int dx[4]={1, 0, -1, 0}, dy[4]={0, 1, 0, -1}; const int fx[8] = {-1, -1, 0, 1, 1, 1, 0, -1}, fy[8] = {0, 1, 1, 1, 0, -1, -1, -1}; const int N = 2e5 + 10; int e[N],ne[N],h[N],id[N],idx; int dep[N],fa[N][21]; int n,m; int s[N]; int ans = -1; void add(int a ,int b , int c) { e[idx] = b, id[idx] = c, ne[idx] = h[a] , h[a] = idx ++; } void dfs(int u , int Fa) { dep[u] = dep[Fa] + 1; fa[u][0] = Fa; for(int k = 1 ; k <= 20 ; k ++) fa[u][k] = fa[fa[u][k-1]][k-1]; for(int i = h[u] ; ~i ; i = ne[i]) { int j = e[i]; if(j == Fa) continue; dfs(j,u); } } int lca(int a , int b) { if(dep[a] < dep[b]) swap(b,a); for(int k = 20 ; k >= 0 ; k --) { if(dep[fa[a][k]] >= dep[b]) a = fa[a][k]; } if(a == b) return b; for(int k = 20 ; k >= 0 ; k --) { if(fa[a][k] != fa[b][k]) { a = fa[a][k]; b = fa[b][k]; } } return fa[a][0]; } void dfs2(int u , int Fa) { for(int i = h[u]; ~i ; i = ne[i]) { int j = e[i]; if(j == Fa) continue; dfs2(j,u); s[u] += s[j]; if(s[j] == m) { ans = max(ans,id[i]); } } } void solve() { cin >> n >> m; memset(h,-1,sizeof h); for(int i = 1 ; i < n ; i ++ ) { int a,b; cin >> a >> b; add(a,b,i),add(b,a,i); } dfs(1,0); for(int i = 0 ; i < m ; i ++) { int a, b; cin >> a >> b; s[a] ++ ; s[b] ++ ; s[lca(a,b)] -= 2 ; } dfs2(1,0); cout << ans ; } int main() { ios::sync_with_stdio(false); cin.tie(nullptr); int t = 1; //cin >> t; while(t--) solve(); return 0; }
0.0分
2 人评分
C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:
一点编程也不会写的:零基础C语言学练课程
解决困扰你多年的C语言疑难杂症特性的C语言进阶课程
从零到写出一个爬虫的Python编程课程
只会语法写不出代码?手把手带你写100个编程真题的编程百练课程
信息学奥赛或C++选手的 必学C++课程
蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程
手把手讲解近五年真题的蓝桥杯辅导课程
发表评论 取消回复