给定一棵N个节点的树,编号1~N。其中1号节点是根,并且第i个节点的权值是Vi

针对这棵树,小Hi会询问小Ho一系列问题。每次小Hi会指定一个节点x,询问小Ho以x为根的子树中,最小的权值是多少。为了增加难度,小Hi可能随时改变其中每个节点的权值。

你能帮助小Ho准确、快速的回答小Hi的问题吗?

Input

第一行一个正整数N。

第二行N个整数,V1, V2, ... VN

第三行n-1个正整数,第i个数Pi表示第i+1号节点的父结点是第Pi号节点。注意1号节点是根。

第四行一个正整数Q,表示有Q个询问/修改权值。

接下来Q行,每行可能有如下两种输入格式:

1 x u

2 x

第一种表示将第x号节点的权值修改为u

第二种表示询问以第x号节点为根的子树中,最小的权值是多少。

对于30%的数据,1 ≤ N, Q ≤ 1000

对于100%的数据,1 ≤ N, Q ≤ 100000, -109 <= Vi, u <= 109

Output

对于每次询问,输出一个整数表示答案。

Sample Input

12
3 5 -1 -2 9 6 2 8 -10 11 8 10
1 1 1 2 4 2 6 7 7 8 8
10
2 3
2 1
2 6
1 11 -5
1 5 -12
2 6
2 4
2 2
2 1
2 7

Sample Output

-1
-10
6
-5
-5
-12
-12
-10

求子树的最小值。单点更新。显然是dfs序,然后按顺序存入线段树里面,由于一颗子树在线段树里的位置相邻,我们可以利用起线段树。

没必要写树剖。由于今天做的几道XOR题太难,我已经疯了。。。

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int maxn=;
int sa[maxn],Rank[maxn];
int Laxt[maxn],Next[maxn],To[maxn],cnt;
int fa[maxn],a[maxn],sz[maxn],tot=;
int ch[maxn][],Min[maxn];
void add(int u,int v)
{
Next[++cnt]=Laxt[u];
Laxt[u]=cnt;
To[cnt]=v;
}
int dfs(int u)
{
sz[u]=;
for(int i=Laxt[u];i;i=Next[i]){
int v=To[i];
sa[v]=++tot;
Rank[tot]=v;
sz[u]+=dfs(v);
} return sz[u];
}
void update(int Now)
{
Min[Now]=min(Min[Now<<],Min[Now<<|]);
}
void build(int Now,int L,int R)
{
if(L==R) { Min[Now]=a[Rank[L]]; return ;}
int Mid=(L+R)>>;
build(Now<<,L,Mid);
build(Now<<|,Mid+,R);
update(Now);
}
int query(int Now,int L,int R,int l,int r)
{
if(l<=L&&r>=R) return Min[Now];
int Mid=(L+R)>>;
if(r<=Mid) return query(Now<<,L,Mid,l,r);
else if(l>Mid) return query(Now<<|,Mid+,R,l,r);
else return min(query(Now<<,L,Mid,l,Mid),query(Now<<|,Mid+,R,Mid+,r));
}
void change(int Now,int L,int R,int x,int val)
{
if(L==R) { Min[Now]=val;return ;}
int Mid=(L+R)>>;
if(x<=Mid) change(Now<<,L,Mid,x,val);
else change(Now<<|,Mid+,R,x,val);
update(Now);
}
int main()
{
int i,j,n,q,x,y,opt;
scanf("%d",&n);
for(i=;i<=n;i++) scanf("%d",&a[i]);
for(i=;i<=n;i++) scanf("%d",&fa[i]),add(fa[i],i);
sa[]=; Rank[]=; dfs(); build(,,n);
scanf("%d",&q);
for(i=;i<=q;i++){
scanf("%d",&opt);
if(opt==) scanf("%d%d",&x,&y),change(,,n,sa[x],y);
else {
scanf("%d",&x);
printf("%d\n",query(,,n,sa[x],sa[x]+sz[x]-));
}
}
return ;
}

最新文章

  1. WebForm Repeater使用
  2. [vijos P1880]ファーラの力
  3. 搭建三层架构(ASP.NET MVC+EF)
  4. Meisell-Lehmer算法(统计较大数据里的素数)
  5. Using Post_Query Trigger in Oracle Forms
  6. Qt入门(13)——Qt的调用退出
  7. Singleton ——运行时全局唯一对象
  8. 关于使用栈将一般运算式翻译为后缀表达式并实现三级运算的方法及实例(cpp版)
  9. [Codeforces 940E]Cashback
  10. 【CF932E】Perpetual Subtraction(NTT,线性代数)
  11. tomcat的Jsp执行
  12. Unity中调用DLL库
  13. ucli tcl cmd
  14. SpringSecurity-RememberMeAuthenticationFilter的作用
  15. 工具使用-----Jmeter教程 简单的压力测试
  16. Java输入输出技术
  17. js 刷新父页面
  18. 【php】header下载文件后,文件变大的问题;(ob_clean()清空缓存)
  19. JAVA容器-模拟ArrayList的底层实现
  20. Partition List,拆分链表

热门文章

  1. 使用zerorpc踩的第一个坑:
  2. mysql 存储过程时间月份减法
  3. OpenSceneGraph FAQ 【转】
  4. springMVC前后端分离开发模式下支持跨域请求
  5. 转: 写给想成为前端工程师的同学们 (from 360前端团队)
  6. dom4j的xpath查找xml的指定节点
  7. nagios插件之登陆SBC监控电话数
  8. 3.nginx反向代理服务器+负载均衡
  9. 【转】Windows2008上传大文件的解决方法(iis7解决上传大容量文件)
  10. redirect_uri 參数错误的解决的方法