Description

有一棵n个点的结构未知的树,初始时只有1号点是已被访问的。

你可以调用交互库的询问函数explore(x,y),其中x是已访问的点,y是任意点。

它会返回x向y方向走第一步的点,如果该点未被访问,则将其标记为已访问。

你需要实现一个函数,它通过接口得到n和T,需要在T次explore操作内将所有的点标记(也就是说走完这棵树)。

要求最严格的两档数据:

n<=300000,T<=300020,且原树为一条链(1号点不一定是端点)。

n<=300000,T<=5000000

Solution

显然我们要将链的情况分开讨论

考虑这样一个做法,我们将编号随机排列,每次找到排列中第一个尚未被访问的点,从当前所在的链端开始explore,若走到的点是未访问的说明这个点就在这一侧,直接一直扩展到目标点。否则说明这个点在链的另一侧,跳到链的另一端一直扩展到目标点。

这样的出错(即链两边跳)的期望次数是\(\log n\)的

大概是因为每次期望都会消掉某一条链的一半这样。

考虑一棵树怎么做。

我们用一个LCT来维护已经扩展出来的树,将1作为根,每一次访问从1开始,在当前所在的prefer链上二分然后explore,若扩展出的点是未访问点则一直怼下去,否则就修改二分区间(实际上在splay上走),如果不在同一条prefer链上就跳到那一条去。

每次找到目标点就access

注意这里需要尽量保持splay的平衡,每次搞出新点都access一下。

具体可以看代码(有些地方可能比较谜,改一点就差很远)

均摊次数就是\(O(n\log n)\)的

Code

#include <bits/stdc++.h>
#include "rts.h"
#define fo(i,a,b) for(int i=a;i<=b;++i)
#define fod(i,a,b) for(int i=a;i>=b;--i)
#define N 300005
using namespace std; int cnt;
bool bz[N];
namespace LCT
{
int sz[N],r[N],f[N],fn[N],dep[N],t[N][2],li[N],ri[N];
void up(int k)
{
sz[k]=sz[t[k][0]]+sz[t[k][1]]+1;
li[k]=(t[k][0])?li[t[k][0]]:k;
ri[k]=(t[k][1])?ri[t[k][1]]:k;
}
void hb(int p,int x,int y)
{
if(x&&p>=0) t[x][p]=y;
if(y) fn[y]=p,f[y]=x;
}
void rot(int k)
{
int fa=f[k],p=fn[k];
hb(p,fa,t[k][1-p]);
hb(fn[fa],f[fa],k);
hb(1-p,k,fa);
up(fa),up(k);
}
int d[N];
void splay(int k,int x)
{
while(f[k]!=x&&fn[k]!=-1&&f[k]!=0)
{
if(f[f[k]]==x||fn[f[k]]==-1||f[f[k]]==0) rot(k);
else if(fn[k]==fn[f[k]]) rot(f[k]),rot(k);
else rot(k),rot(k);
}
up(k);
}
void access(int k)
{
int r=k;
splay(k,0);
fn[t[k][1]]=-1,t[k][1]=0;
up(k);
while(f[k]!=0)
{
int fa=f[k];
splay(fa,0);
fn[t[fa][1]]=-1,hb(1,fa,k);
up(fa),k=fa;
}
splay(r,0);
}
void link(int x,int y)
{
access(x);f[y]=x,fn[y]=-1,up(y);
}
void fd(int x,int y)
{
splay(x,0);
while(x!=y)
{
cnt++;
int p=explore(x,y);
if(p==ri[t[x][0]]) x=t[x][0];
else if(p==li[t[x][1]]) x=t[x][1];
else
{
if(bz[p]) splay(p,0);
else up(p),link(x,p),bz[p]=1;
x=p;
}
}
access(y);
}
}
using namespace LCT; int de[N];
void solve3(int n,int T)
{
int x=1,y=1;
int i=1;
while(i<=n-1)
{
int w=explore(x,de[i]);
if(bz[w]) swap(x,y),w=explore(x,de[i]),cnt++;
bz[w]=1;
while(w!=de[i]) w=explore(w,de[i]),bz[w]=1,cnt++;
x=w;
while(i<=n-1&&bz[de[i]]) i++;
}
} void play(int n, int T, int dataType)
{
memset(bz,0,sizeof(bz));
bz[1]=sz[1]=dep[1]=1;
up(1);
fo(i,2,n) de[i-1]=i,sz[i]=1;
random_shuffle(de+1,de+n);
if(dataType==3) {solve3(n,T);return;}
int i=1,w=1;
while(i<=n-1)
{
fd(1,de[i]);
while(i<=n-1&&bz[de[i]]) i++;
}
n++,n--;
}

[LibreOJ #2341]【WC2018】即时战略【交互】【LCT】的更多相关文章

  1. [WC2018]即时战略(LCT,splay上二分)

    [UOJ题面]http://uoj.ac/problem/349 一道非常好的与数据结构有关的交互题. 首先先看部分分做法, 一上来我们肯定得钦定一个 \(explore\) 的顺序,直接随机就好. ...

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

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

  3. 「WC2018即时战略」

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

  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. 【WC2018】即时战略

    题目描述 小M在玩一个即时战略(Real Time Strategy)游戏.不同于大多数同类游戏,这个游戏的地图是树形的. 也就是说,地图可以用一个由 n个结点,n?1条边构成的连通图来表示.这些结点 ...

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

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

  9. 「WC2018」即时战略

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

  10. 【Unity3D】使用鼠标键盘控制Camera视角(即时战略类游戏视角):缩近,拉远,旋转

    今天写一个demo,要用到鼠标键盘控制三维视角,因此写了个脚本用于控制. 该脚本可以用于即时战略类游戏的视角,提供了缩进,拉伸,旋转.同时按住鼠标右键不放,移动鼠标可以实现第一人称视角的效果. usi ...

随机推荐

  1. 全球数据库-->基金/管理产品-->分类/行业平均-->开放式分类

    SecID 招募书中所定净费率 换手率% 回报日期(每日) 计价货币 回报-本月以来(每日)计价货币 回报-本季以来(每日)计价货币 回报-本年以来(每日)计价货币 回报-1日(每日)计价货币 回报- ...

  2. 关于Spring Data redis几种对象序列化的比较

    redis虽然提供了对list set hash等数据类型的支持,但是没有提供对POJO对象的支持,底层都是把对象序列化后再以字符串的方式存储的.因此,Spring data提供了若干个Seriali ...

  3. awk基础04-内置函数

        在awk中常用的内置函数大概分为:数值函数.字符函数.时间函数.二进制操作函数.数组函数.自定义函数等. 数值函数   常用的数值函数主要有int.rand.srand.sqrt等.详细如下所 ...

  4. java并发编程实战:第六章----任务执行

    任务:通常是一些抽象的且离散的工作单元.大多数并发应用程序都是围绕"任务执行"来构造的,把程序的工作分给多个任务,可以简化程序的组织结构便于维护 一.在线程中执行任务 任务的独立性 ...

  5. 用 Servlet 进行上载的原理和实现

    Servlet 是用 Java 编写的.协议和平台都独立的服务器端组件,使用请求/响应的模式,提供了一个基于 Java 的服务器解决方案.使用 Servlet 可以方便地处理在 HTML 页面表单中提 ...

  6. wp8.1 SQLite的基本使用

    SQLite是一个轻量级的关系型数据库,正是由于其精悍小巧,在移动端平台被广泛应用,但不适合处理大量数据和批量操作.它的底层是由C语言编写,最初设计是为了应用于嵌入式,占用资源非常低且简单易用,而且绝 ...

  7. VUE 学习笔记 二 生命周期

    1.除了数据属性,Vue 实例还暴露了一些有用的实例属性与方法.它们都有前缀 $,以便与用户定义的属性区分开来 var data = { a: 1 } var vm = new Vue({ el: ' ...

  8. 正则表达式REGEXP

    正则表达式:REGular EXPression, REGEXP 元字符: .: 匹配任意单个字符 []: 匹配指定范围内的任意单个字符 [^]:匹配指定范围外的任意单个字符 字符集合:[:digit ...

  9. django drf 权限permission

    https://www.django-rest-framework.org/api-guide/permissions/#custom-permissions from django.shortcut ...

  10. 使用RazorGenerator和预编译MVC引擎将Razor视图编译成DLL

    Web开发中常常会有跨页面.跨站点.跨项目组的复用模块(界面),最常见的就是如下方所示的Web页面上用于显示登录或用户名的头部模块, 使用ASP.NET MVC开发中,常见的做法是写成部分视图,本文的 ...