Codeforces Round #260 (Div. 1)——Civilization
2024-08-31 09:03:14
- 题意:
n个点,m条边的森林。q次操作。每次操作:1、询问x所在树的直径 2、合并x和y所在的树,使得合并后的直径最小
(1 ≤ n ≤ 3·105; 0 ≤ m < n; 1 ≤ q ≤ 3·105) - 分析:
没有读到图是森林。。。做的好纠结
最開始将每一个树都求一下直径,之后使用并查集处理。每次合并直径:至少是两个树的直径,或者将两个直径的最中间的部分连接求直径
const int MAXN = 310000; int rt[MAXN], ans[MAXN];
VI G[MAXN];
bool vis[MAXN]; void init(int n)
{
REP(i, n)
{
vis[i] = false;
ans[i] = 0;
G[i].clear();
rt[i] = i;
}
} int find(int n)
{
return n == rt[n] ? n : rt[n] = find(rt[n]);
} void merge(int a, int b)
{
int fa = find(a), fb = find(b);
if (fa != fb)
{
rt[fa] = fb;
ans[fb] = max(ans[fb], (ans[fb] + 1) / 2 + (ans[fa] + 1) / 2 + 1);
ans[fb] = max(ans[fa], ans[fb]);
}
} int Max, id;
void dfs(int u, int fa, int dep)
{
vis[u] = true;
REP(i, G[u].size())
{
int v = G[u][i];
if (v != fa)
dfs(v, u, dep + 1);
}
if (dep > Max)
{
Max = dep;
id = u;
}
} int main()
{
int n, m, q;
while (~RIII(n, m, q))
{
init(n + 1);
REP(i, m)
{
int a, b;
RII(a, b);
G[a].push_back(b);
G[b].push_back(a);
int fa = find(a), fb = find(b);
rt[fa] = fb;
}
FE(i, 1, n)
{
if (!vis[i])
{
Max = -1;
dfs(i, -1, 0);
Max = -1;
dfs(id, -1, 0);
ans[find(i)] = Max;
}
}
REP(i, q)
{
int op;
RI(op);
if (op == 2)
{
int a, b;
RII(a, b);
merge(a, b);
}
else
{
int a;
RI(a);
WI(ans[find(a)]);
}
}
}
return 0;
}
最新文章
- C# unmanaged function pointers for iOS
- 【JAVA】Quartz中时间表达式的设置
- atitit.重装系统需要备份的资料总结 o84..
- strcmp的实现
- for循环下九九乘法表
- 加密解密(9)Diffie-Hellman密钥交换协议
- CODEVS 3943 数学奇才琪露诺
- 很受欢迎的Linux笔记(短小精悍)
- Joomla插件汉化小程序
- ASP.Net请求处理机制初步探索之旅 - Part 1 前奏(转)
- CSliderCtrl鼠标点击精确定位
- 10分钟精通SharePoint - SharePoint发展历程
- plsql经验之谈
- Linux定时器工具-crontab 各參数具体解释及怎样查看日志记录
- WingIDE 常用快捷键
- win7 英文版 中文乱码
- thinkphp5 中使用 七牛云 上传图片和文件
- C#定义一个方法的3种形式
- layui 三级菜单
- 好用的SHELL小编程