传送门

DFS重标号+二分

打比赛的时候想了很多方法..DFS序,BFS序,倍增什么的都考虑了一遍,但是几乎要么是可以维护两个区间但是代码复杂度爆炸,要么就是只能维护单一维度的信息。

这道题的具体做法就是先DFS遍历一遍,记一下每个点的出入栈时间。按照每个点的深度排序。这样就可以二分出深度,然后根据入栈的时间可以再在这个深度内二分出合适的区间。范围就是询问节点的出入栈时间。

//CF 570D
//by Cydiater
//2016.10.14
#include <iostream>
#include <cstdlib>
#include <queue>
#include <map>
#include <ctime>
#include <cmath>
#include <map>
#include <cstdio>
#include <string>
#include <cstring>
#include <algorithm>
using namespace std;
#define ll long long
#define up(i,j,n)		for(int i=j;i<=n;i++)
#define down(i,j,n)		for(int i=j;i>=n;i--)
const int MAXN=1e6+5;
const int oo=0x3f3f3f3f;
inline int read(){
	char ch=getchar();int x=0,f=1;
	while(ch>'9'||ch<'0'){if(ch=='-')f=-1;ch=getchar();}
	while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
	return x*f;
}
int LINK[MAXN],len=0,N,M,tim=0,pos[MAXN],Xor[MAXN],lef1,lef2,rig1,rig2;
char s[MAXN];
struct edge{
	int y,next;
}e[MAXN];
struct Node{
	int in,out,dep,id,v;
}a[MAXN];
namespace solution{
	inline void insert(int x,int y){e[++len].next=LINK[x];LINK[x]=len;e[len].y=y;}
	inline bool cmp(Node x,Node y){return x.dep==y.dep?x.in<y.in:x.dep<y.dep;}
	void dfs(int Node,int deep){
		a[Node].in=++tim;a[Node].dep=deep;a[Node].id=Node;
		for(int i=LINK[Node];i;i=e[i].next)dfs(e[i].y,deep+1);
		a[Node].out=++tim;
	}
	void init(){
		N=read();M=read();
		up(i,2,N){
			int Node=read();
			if(i==Node)continue;
			insert(Node,i);
		}
		scanf("%s",s+1);
		up(i,1,N)a[i].v=s[i]-'a';
		dfs(1,1);
		sort(a+1,a+N+1,cmp);
		up(i,1,N){
			pos[a[i].id]=i;
			Xor[i]=((1<<a[i].v)^Xor[i-1]);
		}
	}
	int get1(int lim){//<=
		int leftt=1,rightt=N,mid;
		while(leftt+1<rightt){
			mid=(leftt+rightt)>>1;
			if(a[mid].dep>=lim)	rightt=mid;
			else                leftt=mid;
		}
		if(a[leftt].dep>=lim)		return leftt;
		return rightt;
	}
	int get2(int lim){//>=
		int leftt=1,rightt=N,mid;
		while(leftt+1<rightt){
			mid=(leftt+rightt)>>1;
			if(a[mid].dep<=lim)	leftt=mid;
			else                rightt=mid;
		}
		if(a[rightt].dep<=lim)		return rightt;
		return leftt;
	}
	int get3(int lim){//<=
		int leftt=lef1,rightt=rig1,mid;
		while(leftt+1<rightt){
			mid=(leftt+rightt)>>1;
			if(a[mid].in>=lim)	rightt=mid;
			else                leftt=mid;
		}
		if(a[leftt].in>=lim)		return leftt;
		return rightt;
	}
	int get4(int lim){//>=
		int leftt=lef1,rightt=rig1,mid;
		while(leftt+1<rightt){
			mid=(leftt+rightt)>>1;
			if(a[mid].in<=lim)	leftt=mid;
			else                rightt=mid;
		}
		if(a[rightt].in<=lim)		return rightt;
		return leftt;
	}
	void slove(){
		while(M--){
			int Node=pos[read()],dep=read();
			if(a[Node].dep>=dep){puts("Yes");continue;}
			lef1=get1(dep);rig1=get2(dep);
			lef2=get3(a[Node].in+1);rig2=get4(a[Node].out-1);
			int S=Xor[rig2]^Xor[lef2-1];int flag=0;
			if(lef2>rig2){
				puts("Yes");
				continue;
			}
			up(i,0,25)if((S&(1<<i))==(1<<i)&&flag){
				puts("No");
				flag=-1;
				break;
			}else if((S&(1<<i))==(1<<i)) flag=1;
			if(flag!=-1)puts("Yes");
		}
	}
}
int main(){
	//freopen("input.in","r",stdin);
	using namespace solution;
	init();
	slove();
	return 0;
}

CF570D:Tree Requests的更多相关文章

  1. Codeforces Round #316 (Div. 2) D. Tree Requests dfs序

    D. Tree Requests time limit per test 2 seconds memory limit per test 256 megabytes input standard in ...

  2. CF 570 D. Tree Requests

    D. Tree Requests http://codeforces.com/problemset/problem/570/D 题意: 一个以1为根的树,每个点上有一个字母(a-z),每次询问一个子树 ...

  3. Codeforces 570D TREE REQUESTS dfs序+树状数组 异或

    http://codeforces.com/problemset/problem/570/D Tree Requests time limit per test 2 seconds memory li ...

  4. codeforces 570 D. Tree Requests 树状数组+dfs搜索序

    链接:http://codeforces.com/problemset/problem/570/D D. Tree Requests time limit per test 2 seconds mem ...

  5. codeforces 570 D. Tree Requests (dfs)

    题目链接: 570 D. Tree Requests 题目描述: 给出一棵树,有n个节点,1号节点为根节点深度为1.每个节点都有一个字母代替,问以结点x为根的子树中高度为h的后代是否能够经过从新排序变 ...

  6. Codeforces 570D TREE REQUESTS dfs序+树状数组

    链接 题解链接:点击打开链接 题意: 给定n个点的树.m个询问 以下n-1个数给出每一个点的父节点,1是root 每一个点有一个字母 以下n个小写字母给出每一个点的字母. 以下m行给出询问: 询问形如 ...

  7. 爬虫入门系列(三):用 requests 构建知乎 API

    爬虫入门系列目录: 爬虫入门系列(一):快速理解HTTP协议 爬虫入门系列(二):优雅的HTTP库requests 爬虫入门系列(三):用 requests 构建知乎 API 在爬虫系列文章 优雅的H ...

  8. centos7下报错: import requests ImportError: No module named requests

    在网上扒了一个python脚本,在centos7上执行的时候报错: import requestsImportError: No module named requests 原因是:requests是 ...

  9. 转: python requests的安装与简单运用

    requests是Python的一个HTTP客户端库,跟urllib,urllib2类似,那为什么要用requests而不用urllib2呢? 官方文档中是这样说明的: python的标准库urlli ...

随机推荐

  1. 仿造slither.io第一步:先画条蛇

    前言 最近 slither.io 貌似特别火,中午的时候,同事们都在玩,包括我自己也是玩的不亦乐乎. 好久好久没折腾过canvas相关的我也是觉得是时候再折腾一番啦,所以就试着仿造一下吧.楼主也没写过 ...

  2. “奥特曼攻打小怪兽”java学习打怪升级第一步

    ---恢复内容开始--- 练习:回合制对战游戏:奥特曼和小怪兽进行PK,直到一方的血量为0时结束战斗,输出谁胜利了! 不难看出场景中有两个对象:”奥特曼“这一对象抽象为”Ao"类:     ...

  3. 重叠(Overlapping) NAT

    当内部网络也使用公网注册地址(或者是外网合法地址)时,如果仍使用标准的静态或者动态NAT转换,则可能使得转换的内网地址与外网中合法地址冲突,使数据包又返回到了本地网络,这肯定是不行的.这时我们就要使用 ...

  4. Excel——将内容导出

    using (FileStream fsRead = File.OpenRead("111.xls")) { IWorkbook wk = new HSSFWorkbook(fsR ...

  5. java用字符写字符

    import java.awt.Color; import java.awt.Font; import java.awt.Graphics; import java.awt.GraphicsEnvir ...

  6. myeclipse加入JavaEE

    1. 2. 3. 4.

  7. Android判断Touch为滑动事件还是操作控件

    Android判断Touch为滑动事件还是操作控件 因为在项目中要判断WebView是否处于滚动状态,但它不像ListView有onScrollStateChanged方法来监听,要实现就得手动监听它 ...

  8. A+B

    Problem Description Calculate A + B. Input Each line will contain two integers A and B. Process to e ...

  9. iOS开发小技巧-修改SliderBar指针的样式(牢记这个方法,只能通过代码来修改)

    代码: // 修改进度条的指针图片 [self.progressSlider setThumbImage:[UIImage imageNamed:@"player_slider_playba ...

  10. nodeJS+bootstarp+mongodb整一个TODO小例子

    又是一个简单的小玩意儿, 不过有个大玩意儿,就是nosql的mongodb(文件大小:130M),  你要下载一个mongdodb, 去官方网站下载 安装以后在mongodb的命令行目录下执行 mon ...