先把树剖分了(又是dfs1、dfs2),然后区间求和、区间覆盖即可

难得的1A好(shui)题

——写了那么多题,终于有一道是1A的了,加上上一次连续交了几遍A的程序,我的状态莫名好看啊233

总结:

1.开始用define简化代码

2.写的时候出现了手抖,还不够熟练

3.剖分基本上没有问题,线段树题目欠多

 #include <cstdio>
#include <iostream>
#define lson now*2
#define rson now*2+1
#define mid (l+r)/2
using namespace std;
int cnt=,n,m;
int flag[],t[];
int fa[],size[],son[],bro[],b[],pos[],top[];
int dfs1(int k)
{
size[k]=;
for(int i=son[k];i!=;i=bro[i])
size[k]+=dfs1(i);
return size[k];
}
void dfs2(int k,int to)
{
b[++cnt]=k;
top[k]=to;
int max=son[k];
pos[k]=cnt;
for(int i=son[k];i!=;i=bro[i])
if(size[i]>size[max])
max=i;
if(max==)
return;
dfs2(max,to);
for(int i=son[k];i!=;i=bro[i])
if(i!=max)
dfs2(i,i);
}
void down(int now,int l,int r)
{
if(flag[now]==)
{
flag[lson]=;
flag[rson]=;
t[lson]=mid-l+;
t[rson]=r-mid;
flag[now]=;
}
if(flag[now]==)
{
flag[lson]=;
flag[rson]=;
t[lson]=;
t[rson]=;
flag[now]=;
}
}
int que(int now,int l,int r,int x,int y)
{
if(l==x && r==y) return t[now];
down(now,l,r);
return ((x<=mid)?que(lson,l,mid,x,min(y,mid)):)+((y>mid)?que(rson,mid+,r,max(x,mid+),y):);
}
void set_0(int now,int l,int r,int x,int y)
{
if(l==x && r==y)
{
flag[now]=;
t[now]=;
return;
}
down(now,l,r);
if(x<=mid)
set_0(lson,l,mid,x,min(y,mid));
if(y>mid)
set_0(rson,mid+,r,max(x,mid+),y);
t[now]=t[lson]+t[rson];
}
void set_1(int now,int l,int r,int x,int y)
{
if(l==x && r==y)
{
flag[now]=;
t[now]=r-l+;
return;
}
down(now,l,r);
if(x<=mid)
set_1(lson,l,mid,x,min(y,mid));
if(y>mid)
set_1(rson,mid+,r,max(x,mid+),y);
t[now]=t[lson]+t[rson];
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d",&fa[i]);
fa[i]++;
bro[i]=son[fa[i]];
son[fa[i]]=i;
}
dfs1();
dfs2(,);
scanf("%d",&m);
for(int i=;i<=m;i++)
{
char ch=getchar();
while(ch!='i' && ch!='u')
ch=getchar();
bool b=ch=='i';
int x=;
while(ch<'' || ch>'')
ch=getchar();
while(ch>='' && ch<='')
{
x=x*+ch-'';
ch=getchar();
}
x++;
if(b)
{
int sum=;
for(int i=top[x];;i=top[x])
{
sum+=pos[x]-pos[i]+-que(,,n,pos[i],pos[x]);
set_1(,,n,pos[i],pos[x]);
if(i==)
break;
x=fa[i];
}
printf("%d\n",sum);
}
else
{
printf("%d\n",que(,,n,pos[x],pos[x]+size[x]-));
set_0(,,n,pos[x],pos[x]+size[x]-);
}
}
return ;
}

最新文章

  1. java中字节流与字符流的区别
  2. 更改make/bison的版本
  3. Android使用文件存储数据
  4. 融云官方cordova示例使用指南
  5. poj 1815 Friendship 字典序最小+最小割
  6. POJ 1988 Cube Stacking(带权并查集)
  7. NFS网络文件共享服务
  8. Integer ,==,int 的使用
  9. Coder的利器
  10. MySQL中的事务
  11. HBase跨版本数据迁移总结
  12. Flink入门使用
  13. 使用tomcat发布含有shtml文件的web程序
  14. C# 结构(struct)的特点
  15. PLSQL Developer连接远程oracle配置
  16. transition多个属性同时渐变(left/top)
  17. VMware虚拟机Mac OS X无法调整扩展硬盘大小的解决方案
  18. Spring.net(二)----初探IOC容器
  19. 一步一步部署WPF浏览器应用程序
  20. javaSE阶段中 关于Sring类方法的应用

热门文章

  1. Sql Server 分区演练 【转】
  2. AP(affinity propagation)研究
  3. 不要在控制台上使用 let/const
  4. Put-Me-Down项目Postmortem2
  5. thinkphp表单自动验证
  6. CentOS 安装tomcat 7
  7. iOS开发——高级篇——图片轮播及其无限循环效果
  8. Proj.4坐标系统创建参数
  9. python之路三
  10. 获取FIle路径下所有文件的地址和名称