[蓝桥杯]真题讲解:砍树(DFS遍历、图的存储、树上差分与LCA)

2024-01-25 15:28

本文主要是介绍[蓝桥杯]真题讲解:砍树(DFS遍历、图的存储、树上差分与LCA),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

[蓝桥杯]真题讲解:砍树(DFS遍历、图的存储、树上差分与LCA

  • 一、视频讲解
  • 二、暴力代码
  • 三、正解代码

一、视频讲解

视频讲解
在这里插入图片描述

二、暴力代码

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N = 1e5 + 10;typedef pair<int,int> pii;vector<int>edge[N];int n, m;int w[N];//从每一个边的边权。map<pii, int>id;//存边的编号//s是路径的起点,v是路径的重终点,u表示你当前走到了哪个点。
bool dfs(int s, int u, int father, int v)
{if(u == v){return true;}for(int i = 0; i < edge[u].size(); i ++){int son = edge[u][i];if(son == father)continue;if(dfs(s, son, u, v)){int ID = id[{u, son}];w[ID] ++;return true;}}return false;
}void solve()
{cin >> n >> m;for(int i = 0; i < n - 1; i ++){int x, y; cin >> x >> y;edge[x].push_back(y);edge[y].push_back(x);id[{x, y}] = id[{y, x}] = i;}for(int i = 0; i < m; i ++){int x, y; cin >> x >> y;dfs(x, x, -1, y);}int ans = -1;for(int i = n - 1; i >= 0; i --){if(w[i] == m){ans = i + 1;break;}}cout << ans << endl;
}signed main()
{ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);int t = 1;// cin >> t;while(t--)solve();
}

三、正解代码

//砍树:树上差分 + 最近公共祖先
#include<bits/stdc++.h>
using namespace std;
typedef pair<int, int> pii;
const int N = 1e5 + 10;
int n, m;
int w[N];//记录每个点的点权。map<pii,int>id;
vector<int>edge[N];int siz[N], dep[N], fa[N], son[N], top[N];void dfs1(int u, int father)
{siz[u] = 1, dep[u] = dep[father] + 1;fa[u] = father;for(int i = 0; i < edge[u].size(); i ++){int s = edge[u][i];if(s == fa[u])continue;dfs1(s, u);siz[u] += siz[s];if(siz[son[u]] < siz[s])son[u] = s;}
}void dfs2(int u, int t)
{top[u] = t;if(son[u] == 0)return;dfs2(son[u], t);for(int i = 0; i < edge[u].size(); i ++){int s = edge[u][i];if(s == fa[u] || s == son[u])continue;dfs2(s, s);}
}int lca(int x, int y)
{while(top[x] != top[y]){if(dep[top[x]] < dep[top[y]])swap(x, y);x = fa[dep[x]];}return dep[x] < dep[y] ? x : y;
}void cal_sum(int u, int father)
{for(int i = 0; i < edge[u].size(); i ++){int son = edge[u][i];if(son == father)continue;cal_sum(son, u);w[u] += w[son];}
}void solve()
{cin >> n >> m;for(int i = 1; i <= n - 1; i ++){int x, y; cin >> x >> y;edge[x].push_back(y);edge[y].push_back(x);id[{x, y}] = i;id[{y, x}] = i;}	dfs1(1, 0);dfs2(1, 1);for(int i = 0; i < m; i ++){int a, b; cin >> a >> b;//做树上差分w[a] ++, w[b] ++;w[lca(a, b)] -= 2;}cal_sum(1, 0);int ans = -1;for(int i = 1; i <= n; i ++){if(w[i] == m){int ID = id[{i, fa[i]}];ans = max(ans, ID);}}cout << ans << endl;
}signed main()
{ios::sync_with_stdio(0);cin.tie(0);int t = 1;// cin >> t;while(t--)solve();
}

这篇关于[蓝桥杯]真题讲解:砍树(DFS遍历、图的存储、树上差分与LCA)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/643702

相关文章

异构存储(冷热数据分离)

异构存储主要解决不同的数据,存储在不同类型的硬盘中,达到最佳性能的问题。 异构存储Shell操作 (1)查看当前有哪些存储策略可以用 [lytfly@hadoop102 hadoop-3.1.4]$ hdfs storagepolicies -listPolicies (2)为指定路径(数据存储目录)设置指定的存储策略 hdfs storagepolicies -setStoragePo

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k

poj1330(LCA最近公共祖先)

题意:求最近公共祖先 思路:之前学习了树链剖分,然后我就用树链剖分的一小部分知识就可以解这个题目了,记录每个结点的fa和depth。然后查找时,每次将depth大的结点往上走直到x = y。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring>

poj 3159 (spfa差分约束最短路) poj 1201

poj 3159: 题意: 每次给出b比a多不多于c个糖果,求n最多比1多多少个糖果。 解析: 差分约束。 这个博客讲差分约束讲的比较好: http://www.cnblogs.com/void/archive/2011/08/26/2153928.html 套个spfa。 代码: #include <iostream>#include <cstdio>#i

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d

poj 3050 dfs + set的妙用

题意: 给一个5x5的矩阵,求由多少个由连续6个元素组成的不一样的字符的个数。 解析: dfs + set去重搞定。 代码: #include <iostream>#include <cstdio>#include <set>#include <cstdlib>#include <algorithm>#include <cstring>#include <cm

poj 3169 spfa 差分约束

题意: 给n只牛,这些牛有些关系。 ml个关系:fr 与 to 牛间的距离要小于等于 cost。 md个关系:fr 与 to 牛间的距离要大于等于 cost。 隐含关系: d[ i ] <= d[ i + 1 ] 解析: 用以上关系建图,求1-n间最短路即可。 新学了一种建图的方法。。。。。。 代码: #include <iostream>#include

计算机毕业设计 大学志愿填报系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

🍊作者:计算机编程-吉哥 🍊简介:专业从事JavaWeb程序开发,微信小程序开发,定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事,生活就是快乐的。 🍊心愿:点赞 👍 收藏 ⭐评论 📝 🍅 文末获取源码联系 👇🏻 精彩专栏推荐订阅 👇🏻 不然下次找不到哟~Java毕业设计项目~热门选题推荐《1000套》 目录 1.技术选型 2.开发工具 3.功能

POJ 1364差分约束

给出n个变量,m个约束公式 Sa + Sa+1 + .... + Sa+b < ki or > ki ,叫你判断是否存在着解满足这m组约束公式。 Sa + Sa+1   +   .+ Sa+b =  Sum[a+b] - Sum[a-1]  . 注意加入源点n+1 。 public class Main {public static void main(Strin

ural 1149. Sinus Dances dfs

1149. Sinus Dances Time limit: 1.0 second Memory limit: 64 MB Let  An = sin(1–sin(2+sin(3–sin(4+…sin( n))…) Let  Sn = (…( A 1+ n) A 2+ n–1) A 3+…+2) An+1 For given  N print  SN Input One