[UOJ题面]http://uoj.ac/problem/349

一道非常好的与数据结构有关的交互题。

首先先看部分分做法,

一上来我们肯定得钦定一个 \(explore\) 的顺序,直接随机就好。

当 \(n\) 很小的时候就是直接从 1 号点一路 \(explore\) 过去就好了,这样次数是 \(O(n^2)\) 的。

由于完全二叉树树高是 log 的,所以它实际也能过第二个包。

然后来看一下链的情况,稍加思考我们可以得到这么一个做法:就是维护当前已经 \(explore\) 的点的连续区间的左右端点,这样就只需往两边扩展了

这部分是 \(O(n)\) 的。能通过第三个包。

我们把这个想法移到树上,于是就拿一个LCT来维护已经 \(explore\) 出来的点。

那么我们的问题就是怎么快速找到一个新点他在哪,然后我们写一个 \(Splay\) 上二分。

因为询问得到的是 \(now\) 的邻点,而当前这颗 \(Splay\) 上 \(now\) 的邻点只有前驱和后继,那么只要分三种情况讨论一下下一步去哪里就行:

(1)前驱(2)后继(3)它属于另一个 \(Splay\),直接去它那个根就行。

至于找前驱跟后继的话,可以拿 \(Splay\) 顺便维护出来。(可以参考Code)

一些细节:

为了保证复杂度,每次寻点结束时记得 \(Access\);

找根的时候不能 \(Splay\) 否则过不去 \(\text{Extra Test}\)。

最后把算法三、四合起来就行。

#include "rts.h"
#include "algorithm"
using namespace std; const int N=300005;
int vis[N];
int p[N]; #define lc(x) (ch[x][0])
#define rc(x) (ch[x][1]) int ch[N][2],f[N],L[N],R[N]; int g(int x)
{
return rc(f[x])==x;
} int nrt(int x)
{
return lc(f[x])==x || rc(f[x])==x;
} void up(int x)
{
L[x]=R[x]=x;
if(lc(x)) L[x]=L[lc(x)];
if(rc(x)) R[x]=R[rc(x)];
} void rot(int x)
{
int y=f[x],i=g(x);
if(nrt(y))
ch[f[y]][g(y)]=x;
f[x]=f[y];
ch[y][i]=ch[x][i^1];
f[ch[x][i^1]]=y;
ch[x][i^1]=y;
f[y]=x;
up(y);
} void splay(int x)
{
for(int y=f[x];nrt(x);rot(x),y=f[x])
if(nrt(y))
rot(g(x)==g(y)?y:x);
up(x);
} void access(int x)
{
for(int y=0;x;y=x,x=f[x])
{
splay(x);
rc(x)=y;
}
} int gr(int x)
{
for(;nrt(x);x=f[x]); return x;
} void play(int n, int T, int dataType) {
vis[1]=1;
if(dataType==2)
{
for(int i=2; i<=n; i++)
{
if(vis[i])
continue;
int now=1;
while(now!=i) {
now=explore(now,i);
vis[now]=1;
}
}
}
else if(dataType==3)
{
for(int i=1;i<=n;i++)p[i]=i;
int l,r;
l=r=1;
srand(20030118);
for(int i=1;i<=5*n;i++)
{
int x=rand()%n+1;
int y=rand()%n+1;
swap(p[x],p[y]);
}
for(int i,j=1;j<=n;j++)
{
i=p[j];
if(vis[i])
continue;
int now=explore(l,i);
if(!vis[now])
{
vis[now]=1;
while(now!=i) {
now=explore(now,i);
vis[now]=1;
}
l=i;
}
else
{
now=explore(r,i);
vis[now]=1;
while(now!=i) {
now=explore(now,i);
vis[now]=1;
}
r=i;
}
}
}
else
{
for(int i=1;i<=n;i++)p[i]=i;
srand(20030118);
for(int i=1;i<=5*n;i++)
{
int x=rand()%n+1;
int y=rand()%n+1;
swap(p[x],p[y]);
}
for(int i,j=1;j<=n;j++)
{
i=p[j];
if(vis[i])
continue;
int now=gr(1);
while(now!=i)
{
while(1)
{
int t=explore(now,i);
if(t==R[lc(now)]) now=lc(now);
else if(t==L[rc(now)]) now=rc(now);
else {
if(!vis[t])
vis[t]=1,f[t]=now;
now=gr(t);
break;
}
}
}
access(now);
}
}
}

[WC2018]即时战略(LCT,splay上二分)的更多相关文章

  1. [WC2018]即时战略——动态点分治(替罪羊式点分树)

    题目链接: [WC2018]即时战略 题目大意:给一棵结构未知的树,初始时除1号点其他点都是黑色,1号点是白色,每次你可以询问一条起点为白色终点任意的路径,交互库会自动返回给你这条路径上与起点相邻的节 ...

  2. 「WC2018即时战略」

    「WC2018即时战略」 题目描述 小 M 在玩一个即时战略 (Real Time Strategy) 游戏.不同于大多数同类游戏,这个游戏的地图是树形的.也就是说,地图可以用一个由 \(n\) 个结 ...

  3. 【UOJ349】【WC2018】即时战略 LCT 动态点分治

    这是一道交互题 题目大意 有一棵\(n\)个点的树.最开始\(1\)号点是白的,其他点是黑的. 每次你可以执行一个操作:\(explore(x,y)\).要求\(x\)是一个白点.该函数会返回从\(x ...

  4. WC2018 即时战略

    交互题 一棵树,一开始只有 1 号点是已知的,其他的都是未知的,你可以调用函数 explore(x,y) ,其中 x 必须是已知的,函数会找到 x 到 y 路径上第二个点,并把它标成已知,求最小步数使 ...

  5. 【UOJ#349】[WC2018] 即时战略

    题目链接 题意 一开始已知一号点. 每次可以选定一个已知点和一个未知点,然后交互库会返回从已知点出发到达未知点路径上的第二个点. 要求在有限步之内知道每一个点. 次数要求: 链的情况要求 \(O(n) ...

  6. loj2341「WC2018」即时战略(随机化,LCT/动态点分治)

    loj2341「WC2018」即时战略(随机化,LCT/动态点分治) loj Luogu 题解时间 对于 $ datatype = 3 $ 的数据,explore操作次数只有 $ n+log n $ ...

  7. [LibreOJ #2341]【WC2018】即时战略【交互】【LCT】

    Description 有一棵n个点的结构未知的树,初始时只有1号点是已被访问的. 你可以调用交互库的询问函数explore(x,y),其中x是已访问的点,y是任意点. 它会返回x向y方向走第一步的点 ...

  8. 【WC2018】即时战略(动态点分治,替罪羊树)

    [WC2018]即时战略(动态点分治,替罪羊树) 题面 UOJ 题解 其实这题我也不知道应该怎么确定他到底用了啥.只是想法很类似就写上了QwQ. 首先链的部分都告诉你要特殊处理那就没有办法只能特殊处理 ...

  9. 「WC2018」即时战略

    「WC2018」即时战略 考虑对于一条链:直接随便找点,然后不断问即可. 对于一个二叉树,树高logn,直接随便找点,然后不断问即可. 正解: 先随便找到一个点,问出到1的路径 然后找别的点,考虑问出 ...

随机推荐

  1. An easy problem(位运算)

    As we known, data stored in the computers is in binary form.(数据以二进制形式存储于电脑之中.)The problem we discuss ...

  2. vs2015编译各种库

    1.编译libcurl https://www.jianshu.com/p/f82d3d18da93

  3. C# LINQ GroupBy

    一.先准备要使用的类: 1.Person类: class Person { public string Name { set; get; } public int Age { set; get; } ...

  4. VC++编译选项

    -优化- /O1 最小化空间 minimize space /Op[-] 改善浮点数一致性 improve floating-pt consistency /O2 最大化速度 maximize spe ...

  5. Java学习笔记(十一)面向对象---多态

    多态的体现 父类的引用指向了自己的子类对象. 父类的引用也可以接受自己的子类对象. 代码体现 abstract class Animal { public abstract void eat(); } ...

  6. php设计模式之工厂方法实例代码

    实现不修改原代码,扩展新功能 <?php header("Content-type:text/html;charset=utf-8"); /** * db接口 * 实现连接数 ...

  7. span标签间距

    最近在做的一个项目里面碰到这么一个问题: <p> <span>块1</span> <span>块2</span> </p> 在“ ...

  8. 051_switch语句的使用 052_while循环详解 053_for循环详解_dowhile简介 054_嵌套循环_循环相关练习

    051_switch语句的使用 package testmode2;/** * 测试switch语句 * 遇到多值判断的时候,使用switch.当然,switch完全可以使用ifelseifelse代 ...

  9. 问题 B: 奇怪的电梯

    问题 B: 奇怪的电梯 时间限制: 1 Sec  内存限制: 128 MB[命题人:admin] 题目描述 大楼的每一层楼都可以停电梯,而且第i层楼(1<=i<=N)上有一个数字Ki(0& ...

  10. Eqaulize Prices

    There are n products in the shop. The price of the ii-th product is aiai. The owner of the shop want ...