题目链接

\(Description\)

  有一棵高度为\(h\)的满二叉树,点从\(1\)到\(2^h-1\)编号(无序)。每次你可以询问一个点的编号,交互库会返回其所有邻接点的编号。你需要在\(16\)次询问内确定这棵树根节点的编号。

  \(h\leq 7\)。

\(Solution\)

  考虑随便问一个点,然后任意找个相邻点走。这样如果不往回走,最差情况下是一直走到一个叶子,这样找走两遍,扩展出一条叶子到叶子的链,就可以往上扩展了。这样最多扩展\(1+2+\ldots+7=28\)个点,但是确定根节点就够了,即\(21\)个。

  还是不行。在深度比较浅时代价会比较高,但是深度浅了我们离根节点就更近。所以在离根足够近(距离为\(2\))直接BFS。这样代价为\(10+1+2+4=17\),但是最后一个点不需要查知道了,代价为\(16\)。

  思路很好理解,但是代码好难写啊。。弃疗了。参考个吧。orz\(yanQval\).

  要对初始点DFS两次,不管路径如何,我们记下两条路径经过点数\(c_1,c_2\),其深度就是\(\frac{c_1+c_2}{2}+1\)。如果有一次是向根节点延伸(\(c_1\neq c_2\)),就可以直接跳到经过路径上最靠近根的点。

  之后保证每次向上走,用之前的深度和新路径的点数同样可以跳。最后手动BFS。

#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
const int N=150; int h,dgr[N],son[N][3],A1[N],A2[N];
bool vis[N]; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
#define Check(x) if(dgr[x]==2) return x
inline void Query(int x)
{
vis[x]=1;
printf("? %d\n",x), fflush(stdout);
dgr[x]=read();
for(int i=0; i<dgr[x]; ++i) son[x][i]=read();
}
inline int Step(int x)
{
for(int i=0; i<dgr[x]; ++i) if(!vis[son[x][i]]) return son[x][i];
return son[x][0];
}
int Solve()
{
memset(vis,0,sizeof vis);
int h=read(), x=rand()%((1<<h)-1)+1, dep;
Query(x); Check(x);
if(dgr[x]==1) dep=1;
else
{
int cnt1=0, cnt2=0;
for(int v=Step(x); ; v=Step(v))
{
Query(v), A1[++cnt1]=v; Check(v);
if(dgr[v]==1) break;
}
for(int v=Step(x); ; v=Step(v))
{
Query(v), A2[++cnt2]=v; Check(v);
if(dgr[v]==1) break;
}
dep=(cnt1+cnt2>>1)+1;
if(cnt1>cnt2) x=A1[cnt1-dep+1];
else if(cnt1<cnt2) x=A2[cnt2-dep+1];
}
for(int cnt=0; dep<4/*not 5*/; cnt=0)
{
for(int v=Step(x); ; v=Step(v))
{
Query(v), A1[++cnt]=v; Check(v);
if(dgr[v]==1) break;
}
dep=dep+cnt+1>>1, x=A1[cnt-dep+1];
}
int a,b,c,d,e;
if(dep<h)
{
x=Step(x), Query(x); Check(x);
}
if(dep<h-1)
{
a=Step(x), Query(a); Check(a);
b=Step(x), Query(b); Check(b);
}
if(dep<h-2)
{
c=Step(a), Query(c); Check(c);
d=Step(a), Query(d); Check(d);
e=Step(b), Query(e); Check(e);
return Step(b);
}
return x;
} int main()
{
for(int T=read(); T--; printf("! %d\n",Solve()),fflush(stdout));
return 0;
}

Good Bye 2016 F.New Year and Finding Roots(交互)的更多相关文章

  1. codeforces Good bye 2016 E 线段树维护dp区间合并

    codeforces Good bye 2016 E 线段树维护dp区间合并 题目大意:给你一个字符串,范围为‘0’~'9',定义一个ugly的串,即串中的子串不能有2016,但是一定要有2017,问 ...

  2. 2016-2017 ACM-ICPC Southwestern European Regional Programming Contest (SWERC 2016) F dfs序+树状数组

    Performance ReviewEmployee performance reviews are a necessary evil in any company. In a performance ...

  3. Good Bye 2016

    A - New Year and Hurry (water) #include <bits/stdc++.h> using namespace std; int main() { ]; ; ...

  4. CF IndiaHacks 2016 F Paper task 后缀数组

    题目链接:http://codeforces.com/problemset/problem/653/F 大意是给出一个只包含'('和')'的括号串,求有多少不同的子串是合法的括号串 解法:对于每一个后 ...

  5. 2016-2017 ACM-ICPC East Central North America Regional Contest (ECNA 2016) F 区间dp

    Problem F Removal GameBobby Roberts is totally bored in his algorithms class, so he’s developed a li ...

  6. Good Bye 2015 F - New Year and Cleaning

    F - New Year and Cleaning 这题简直是丧心病狂折磨王.. 思路:容易想到这样一个转换,把整个矩形一起移动,矩形移出去的时候相当于一行或者一列. 为了优化找到下一个消去的点,我先 ...

  7. Good Bye 2014 F - New Year Shopping

    F - New Year Shopping 对于一种特殊的不可逆的dp的拆分方法.. 也可以用分治写哒. #include<bits/stdc++.h> #define LL long l ...

  8. April Fools Day Contest 2016 F. Ace It!

    F. Ace It! 题目连接: http://www.codeforces.com/contest/656/problem/F Description Input The only line of ...

  9. Good Bye 2016 A. New Year and Hurry【贪心/做题目每道题花费时间按步长为5等差增长,求剩余时间够做几道题】

    A. New Year and Hurry time limit per test 1 second memory limit per test 256 megabytes input standar ...

随机推荐

  1. 安装Python和Anaconda

    安装Python和Anaconda 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.安装anaconda anaconda包括了Python的集成开发环境. 1.打开下载的网站 ...

  2. spring boot 事务配置

    事务的作用这里不细说,相信很多人也在工作中使用过. 那么在spring-boot是如何配置事务的,事实上非常简便. 直接贴代码吧. 首先配置数据源  myqlDataSource,这个就不说了.之前的 ...

  3. Knowledge Tracing -- 基于贝叶斯的学生知识点追踪(BKT)

    目前,教育领域通过引入人工智能的技术,使得在线的教学系统成为了智能教学系统(ITS),ITS不同与以往的MOOC形式的课程.ITS能够个性化的为学生制定有效的 学习路径,通过根据学生的答题情况追踪学生 ...

  4. Spark记录-Scala集合

    Scala列表 Scala列表与数组非常相似,列表的所有元素都具有相同的类型,但有两个重要的区别. 首先,列表是不可变的,列表的元素不能通过赋值来更改. 其次,列表表示一个链表,而数组是平的. 具有类 ...

  5. ASP.NET生成二维码

    下面使用ThoughtWorks.QRCode.dll这个类库,实现生成二维码 使用时需要增加:下面三个命名空间 using ThoughtWorks.QRCode.Codec; using Thou ...

  6. v140平台工具集与v110工具集选择

    今天在编译用vs2012编译C++动态库提示:error MSB8020: The builds tools for v140_xp (Platform Toolset = 'v140_xp') ca ...

  7. jquery的几种ajax方式对比

    jquery的几种ajax方式对比 jquery的ajax方式有如下几种: 1.   $.post(url,params,callback); 2.   $.getJSON(url,params,ca ...

  8. python基础-装饰器,生成器和迭代器

    学习内容 1.装饰器 2.生成器 3.迭代器 4.软件目录结构规范 一:装饰器(decorator) 1.装饰器定义:本质就是函数,用来装饰其他函数,即为其他函数添加附加功能. 2.装饰器原则:1)不 ...

  9. HTML5 drawImage性能问题

    最近做游戏发现,为了节省HTTP请求,把很多组动画放到了一张图片上,也就是传说中的雪碧图,但是用drawImage从图片上截取一部分绘制在canvas上时会非常慢. 我的图片尺寸为5000*7000 ...

  10. ThinkPHP中的统计查询方法

    • count() 表示查询表中总的记录数 • max() 表示查询某个字段的最大值 • min() 表示查询某个字段的最小值 • avg() 表示查询某个字段的平均值 • sum() 表示求出某个字 ...