首先看两条链怎么合并,贪心可得是从大到小取max,多条链同理

所以dfs合并子树的大根堆即可,注意为了保证复杂度,合并的时候要合并到最长链上,证明见长链剖分

#include<iostream>
#include<cstdio>
#include<queue>
using namespace std;
const int N=200005;
int n,a[N],h[N],cnt,id[N],tot,p[N];
long long ans;
priority_queue<int>q[N];
struct qwe
{
int ne,to;
}e[N<<1];
int read()
{
int r=0,f=1;
char p=getchar();
while(p>'9'||p<'0')
{
if(p=='-')
f=-1;
p=getchar();
}
while(p>='0'&&p<='9')
{
r=r*10+p-48;
p=getchar();
}
return r*f;
}
void add(int u,int v)
{
cnt++;
e[cnt].ne=h[u];
e[cnt].to=v;
h[u]=cnt;
}
void dfs(int u)
{//cerr<<u<<endl;
id[u]=++tot;
for(int i=h[u];i;i=e[i].ne)
{
dfs(e[i].to);
if(q[id[e[i].to]].size()>q[id[u]].size())
swap(id[e[i].to],id[u]);
int len=q[id[e[i].to]].size();
for(int j=1;j<=len;j++)
{
p[j]=max(q[id[e[i].to]].top(),q[id[u]].top());
q[id[e[i].to]].pop();
q[id[u]].pop();
}
for(int j=1;j<=len;j++)
q[id[u]].push(p[j]);
}
q[id[u]].push(a[u]);
}
int main()
{
n=read();
for(int i=1;i<=n;i++)
a[i]=read();
for(int i=2;i<=n;i++)
{
int x=read();
add(x,i);
}
dfs(1);
while(!q[id[1]].empty())
ans+=q[id[1]].top(),q[id[1]].pop();
printf("%lld\n",ans);
return 0;
}

最新文章

  1. Python学习笔记——字典
  2. 关于Xcode6创建的工程在Xcode5打开
  3. 洛谷P3128 [USACO15DEC]最大流Max Flow [树链剖分]
  4. Order Independent Transparency
  5. CentOS 6.5 3.0.4安装agentd
  6. Oracle 增加修改删除字段与添加注释
  7. contenteditable
  8. 在xml文件中写入&amp;符号时需要对其进行转义
  9. webstorm卡、闪退以及win10中jdk配置
  10. JVM笔记-temp
  11. HTML&amp;CSS基础学习笔记1.17-表格的头部与尾部
  12. 新浪微博iOS示例,登录,获取个人信息
  13. Ueditor的配置及使用
  14. python3中socket套接字的编码问题解决
  15. python爬虫(1)——urllib包
  16. XSS,CSRF,Cookie防劫持的处理
  17. maven项目自动创建src/main/resources等四个资源文件夹
  18. laravel 列表搜索查询(when,with用法以及关联图像id处理图像路径)
  19. JAVA 实验报告
  20. lucene4之后的近实时搜索实现

热门文章

  1. HTML制作练习
  2. 初解C#类、结构、弱引用
  3. HDU 6068 Classic Quotation KMP+DP
  4. 基于mqtt协议实现手机位置跟踪
  5. 常用SQL备忘录
  6. ivy
  7. react native 中的redux
  8. html5--6-60 阶段练习7-下拉菜单
  9. 通过Toad工具查看dmp里面的表
  10. codeforces 691A A. Fashion in Berland(水题)