博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
CF1174F Ehab and the Big Finale 树分治
阅读量:4664 次
发布时间:2019-06-09

本文共 1037 字,大约阅读时间需要 3 分钟。

 题意: 交互题,给出一棵树,1为树根,有一点x为目标点。

有两种操作 d a: 询问a到x的距离,s a:询问从a到x的路径上第二个结点的编号,注意a必须要是x的祖先结点。

需要在36次询问内给出x结点

思路:先从树根开始一次dfs,sz数组维护以i为根节点的树拥有的结点数(包括i本身)。 询问一次d 1,得到x的深度depx

然后从树根开始,每次经过sz值最大的子节点,直到该路径的终点v,并询问d v 得到dist(x,v).

因为我们现在知道depv,depx,dist(x,v),那么我们可以知道y=lca(v,x)的深度depy, dist(x,v)=depx+depv2depy

因为该路径上的所有点深度都不同,所以我们可以得到y点的位置,询问s y,得到从y到x路径上第二个结点的编号。

进行下一次dfs,当depv==depx时,x的位置便是该次dfs上路径的唯一点。于是我们得到x的位置

 

#include
using namespace std; const int maxn=2e5+5; vector
v[maxn],h; int sz[maxn],dep[maxn],depx; int pre(int node,int p){ sz[node]=1; for(int u:v[node]){ if(u!=p){ dep[u]=dep[node]+1; sz[node]+=pre(u,node); } } return sz[node]; } int query(char c,int u){ printf("%c %d\n",c,u); fflush(stdout); int ans; scanf("%d",&ans); return ans; } void get(int node){ h.push_back(node); int big=-1; for(int u:v[node]){ if(sz[node]>sz[u]&&(big==-1||(sz[u]>sz[big]))) big=u; } if(big!=-1) get(big); } int dfs(int node){ h.clear(); get(node); int depy=(depx+dep

转载于:https://www.cnblogs.com/hanker99/p/10981902.html

你可能感兴趣的文章
如何进行Apache虚拟机设置
查看>>
【水滴石穿】报错解决不了
查看>>
Qt之Tab键实现(自由切换焦点)
查看>>
Codeforces 920F. SUM and REPLACE / bzoj 3211 花神游历各国
查看>>
Cocos2d-x 3.2 学习笔记(七)Scene And Transition
查看>>
Re:JavaScript
查看>>
ajax 200 4 parseerror的一个问题
查看>>
面试题之实现系统函数系列一:实现memmove函数
查看>>
数据结构:可持久化并查集
查看>>
基于UDP协议的Socket通信
查看>>
linux安装配置Redis,Swoole扩展
查看>>
C语言-02基本运算
查看>>
轻巧快速的JSON工具--fastJSON
查看>>
生男生女预测软件,千人验证无误
查看>>
js call()和apply()方法的区别和用法详解
查看>>
Android之Service
查看>>
HDU 2795 Billboard 解题报告
查看>>
多线程——newCachedThreadPool线程池
查看>>
日志传输与清除脚本
查看>>
maven小知识点
查看>>