题目

分析

首先预处理出每个点的优先级,当有一个人进入时,一定会走到优先级最大的空房间中。

把所有空的房间扔到一个堆中,按优先级大小维护这个堆。

答案怎么求就不说了,很容易想到,就只讲操作吧。

对于第一种操作,我们就将一个一个扔进优先级最大的房间(即堆顶),因为堆顶的房间有人,所以将堆顶的房间踢出堆。

对于第二种操作,当一个房间空了,一定是从它的父亲房间补上来。

倍增将深度最大的空了的房间的祖宗取出来,因为它的祖宗们一定一个接一个往下走一个位置。

也就是说,深度最大的空了的房间的祖宗人的人会走掉,那么将这个房间加入堆。

#include <cmath>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
#include <queue>
const int maxlongint=2147483647;
const int mo=1000000007;
const int N=110005;
using namespace std;
struct ddx
{
int x,y;
}a[N*2];
int last[N*4],next[N*4],to[N*4],sum,n,m,t,ans,b[N*2],tot,belong[N*2];
int heap[N*2],th[N*2],g[N*2][20],tot1,deep[N*2];
bool bz[N*2];
bool cmp(ddx x,ddx y)
{
return x.x<y.x || (x.x==y.x && x.y<y.y);
}
void bj(int x,int y)
{
next[++tot]=last[x];
last[x]=tot;
to[tot]=y;
}
int up(int x)
{
while(th[heap[x/2]]>th[heap[x]] && x!=1)
{
belong[heap[x]]=x/2;
belong[heap[x/2]]=x;
heap[x]=heap[x]^heap[x/2];
heap[x/2]=heap[x]^heap[x/2];
heap[x]=heap[x]^heap[x/2];
x/=2;
}
}
int down(int x)
{
while((th[heap[x]]>th[heap[x*2]] || th[heap[x]]>th[heap[x*2+1]]) && x*2<=sum)
{
if(x*2+1>sum && th[heap[x]]<th[heap[x*2]]) break;
if(th[heap[x*2]]<th[heap[x*2+1]] || x*2+1>sum)
{
belong[heap[x]]=x*2;
belong[heap[x*2]]=x;
heap[x]=heap[x]^heap[x*2];
heap[x*2]=heap[x]^heap[x*2];
heap[x]=heap[x]^heap[x*2];
x=x*2;
}
else
{
belong[heap[x]]=x*2+1;
belong[heap[x*2+1]]=x;
heap[x]=heap[x]^heap[x*2+1];
heap[x*2+1]=heap[x]^heap[x*2+1];
heap[x]=heap[x]^heap[x*2+1];
x=x*2+1;
}
}
}
int insert(int x)
{
heap[++sum]=x;
belong[x]=sum;
up(sum);
}
int cut()
{
belong[heap[sum]]=1;
belong[heap[1]]=0;
heap[1]=heap[sum--];
heap[sum+1]=0;
down(1);
}
void dg(int x)
{
deep[x]=deep[g[x][0]]+1;
for(int i=b[x];a[i].x==x;i++)
{
if(a[i].y!=g[x][0])
{
g[a[i].y][0]=x;
bj(x,a[i].y);
dg(a[i].y);
}
}
th[x]=++tot1;
insert(x);
}
int lca(int x)
{
int num=x;
for(int i=log2(n);i>=0;i--)
if((!belong[g[x][i]]) && g[x][i])
x=g[x][i];
if((!belong[g[x][0]]) && g[x][0]) x=g[x][0];
printf("%d\n",deep[num]-deep[x]);
return x;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n-1;i++)
{
scanf("%d%d",&a[i].x,&a[i].y);
a[n+i-1].x=a[i].y;
a[n+i-1].y=a[i].x;
}
sort(a+1,a+n*2-1,cmp);
for(int i=1;i<=n*2-2;i++)
if(a[i].x!=a[i-1].x) b[a[i].x]=i;
dg(1);
for(int i=1;i<=log2(n);i++)
for(int j=1;j<=n;j++)
g[j][i]=g[g[j][i-1]][i-1];
for(int i=1;i<=m;i++)
{
int x,y;
scanf("%d%d",&x,&y);
if(x==1)
{
for(int j=1;j<=y;j++)
{
if(j==y)
printf("%d\n",heap[1]);
cut();
}
}
else
{
insert(lca(y));
}
}
}

【NOIP2016提高A组五校联考1】排队的更多相关文章

  1. NOIP2016提高A组五校联考4总结

    坑爹的第一题,我居然想了足足3个小时,而且还不确定是否正确. 于是,我就在这种情况下心惊胆跳的打了,好在ac了,否则就爆零了. 第二题,树形dp,本来差点就想到了正解,结果时间不够,没打完. 第三题, ...

  2. 【NOIP2016提高A组五校联考4】square

    题目 分析 首先,设\(f_{i,j}\)表示最大的以(i,j)为左下角的正方形的边长. 转移显然,\(f_{i,j}=\max(f_{i-1,j},f_{i,j-1},f_{i-1,j-1})+1\ ...

  3. 【NOIP2016提高A组五校联考4】label

    题目 题目 20%算法 设\(f_{i,j}\)表示第i个节点选了j这个权值的方案数. 显然转移方程为,\[f_{i,j}=\Pi_{v=son(i)}(\sum_{k=1}^{j-k}f_{v,k} ...

  4. 【NOIP2016提高A组五校联考4】ksum

    题目 分析 发现,当子段[l,r]被取了出来,那么[l-1,r].[l,r+1]一定也被取了出来. 那么,首先将[1,n]放入大顶堆,每次将堆顶的子段[l,r]取出来,因为它是堆顶,所以一定是最大的子 ...

  5. NOIP2016提高A组五校联考3总结

    第一题,本来一开始就想到了数位dp,结果脑残地打了十几个转移方程,总是调试不出来,一气之下放弃了. 调第一题几乎调了整节比赛,第二第三都没它. 第二题连边找联通块. 第三题题解都打了三页,看都不想看. ...

  6. 【NOIP2016提高A组五校联考2】tree

    题目 给一棵n 个结点的有根树,结点由1 到n 标号,根结点的标号为1.每个结点上有一个物品,第i 个结点上的物品价值为vi. 你需要从所有结点中选出若干个结点,使得对于任意一个被选中的结点,其到根的 ...

  7. 【NOIP2016提高A组五校联考2】running

    题目 小胡同学是个热爱运动的好孩子. 每天晚上,小胡都会去操场上跑步,学校的操场可以看成一个由n个格子排成的一个环形,格子按照顺时针顺序从0 到n- 1 标号. 小胡观察到有m 个同学在跑步,最开始每 ...

  8. 【NOIP2016提高A组五校联考2】string

    题目 给出一个长度为n, 由小写英文字母组成的字符串S, 求在所有由小写英文字母组成且长度为n 且恰好有k 位与S 不同的字符串中,给定字符串T 按照字典序排在第几位. 由于答案可能很大,模10^9 ...

  9. NOIP2016提高A组五校联考2总结

    第一题用组合数各种乱搞,其恶心程度不一般.搞了很久才调对,比赛上出了一点bug,只拿了30分. 第二题我乱搞得出个错误的结论,本来自信满满60分,结果爆零了. 第三题,树形dp,在一开始的时候想到了, ...

  10. NOIP2016提高A组五校联考1总结

    第一题二分,在比赛上明明想到的方法,结果考虑的时候似乎漏了什么,被否决掉了. 只打了个水法,10分. 第二题,最长不上升子序列,原题,类似的题目做过两道,直接搞定. 第三题,一开始想了一种通过在树上打 ...

随机推荐

  1. ubuntu16.04离线安装nvidia-docker2

    目前需要离线对ubuntu 进行封装docker环境 在熟悉docker环境过程中,有网络条件下,还处于懵逼状态 离线安装…… 大佬救救我 来了: 首先下载安装docker最新版 我的OS是Ubunt ...

  2. USACO4.3 Buy Low, Buy Lower【简单dp·高精度】

    如果没有方案数的话,这道题水的不得了,裸的最长下降子序列. 但是它有方案数,所以... 一个是方案数的求法: 设$f[i]$是以$a[i]$结尾的最长下降子序列的长度,可以$n^2$$dp$出答案 如 ...

  3. SSM004/工作内容

    一.java执行sql脚本 参考博客:java调用SQL脚本执行的方案 1.Service层代码:目的随spring容器启动即执行 //Service层代码 @Component public cla ...

  4. flask 必知必会

    在局域网中让其它电脑访问我的网站 from flask import Flask app = Flask(__name__) @app.route('/') def hello_world(): re ...

  5. linux c++ 实现http请求

    main.cpp #include HttpReq.h #include <string.h> int main(void) { HttpRequest* Http; char http_ ...

  6. linux-查询某软件的安装的目录

    eg:jenkins\\\ rpm -ql jenkins 安装目录/var/lib/jenkins 配置文件 /etc/sysconfig/jenkins 日志目录 /var/log/jenkins ...

  7. vue-router的query和params的区别

    vue-router的query和params的区别 首先简单来说明一下$router和$route的区别 $router为VueRouter实例,想要导航到不同url,则使用$router.push ...

  8. synchronize和lock的区别 & synchionzie与volatile的区别

    synchronized与Lock的区别 https://www.cnblogs.com/iyyy/p/7993788.html Lock和synchronized和volatile的区别和使用 ht ...

  9. AQS之Condition

    一.引言 一般我们在使用锁的Condition时,我们一般都是这么使用,以ReentrantLock为例, ReentrantLock lock = new ReentrantLock(); Cond ...

  10. CF Gym102028G Shortest Paths on Random Forests

    传送门 这题要求的期望,就是总权值(所有不在同一个连通块点对的贡献+同一连通块点对的贡献)/总方案(森林个数) 先求森林个数,森林是由一堆树组成的,而根据purfer序列,一棵\(n\)个点的有标号的 ...