BZOJ

LOJ

洛谷


\(Kelin\)写的挺清楚的...

要求如果\(a_{p_j}=p_k\),\(k\lt j\),可以理解为\(k\)要在\(j\)之前选。

那么对于给定的\(a_j=k\),我们可以连边\(k\to j\)建一张图。如果图有环,那么无解;否则这是一棵以\(0\)为根的树。

限制就变成了,选一个点前必须要选其父亲;如果第\(k\)个选点\(i\),\(i\)的贡献是\(k\cdot w_i\)。

然后怎么做呢...

假设当前最小的数是\(x\),那么如果\(fa[x]\)选过,现在一定选\(x\);否则如果有一次选了\(fa[x]\),下一次一定选\(x\)。

考虑到按顺序选点实际是构成一个序列,且上面的\(x\)一定和\(fa[x]\)在序列中挨在一起。所以考虑把\(x,fa[x]\)合并成一个序列,再考虑其它的。

之前是比点权,现在我们要比较序列与序列之间谁放在后面更优。

考虑两个序列\(a,b\),设\(a\)中有\(t_1\)个点,\(b\)中有\(t_2\)个点,假设现在已经选了\(k\)个点,那么\(ab,ba\)的贡献分别为$$W_{ab}=\sum_{i=1}{t_1}(k+i)w_{a_i}+\sum_{i=1}{t_2}(k+i+t_1)w_{b_i}\W_{ba}=\sum_{i=1}{t_2}(k+i)w_{b_i}+\sum_{i=1}{t_1}(k+i+t_2)w_{a_i}\W_{ab}-W_{ba}=t_1\sum w_{b_i}-t_2\sum w_{a_i}$$

那么\(W_{ab}>W_{ba}\Rightarrow \frac{\sum w_{b_i}}{t_2}>\frac{\sum w_{a_i}}{t_1}\),即平均数更小的放到前面更优。

所以我们就可以拿堆维护了,每次取出最小的,把它与父节点所在序列合并。这里可以用带修改堆修改父节点的状态,也可以打个标记删掉父节点的状态(其实判一下\(sz\)是否等于当前状态下的\(sz\)就可以了)。

答案就在一个序列被放到后面的时候统计就可以了。

另外堆也可以用\(STL\)里的\(heap\)。只需要#include <algorithm>

常数应该是比priority_queue小一些的?可能是我写的丑...


priority_queue

//30268kb	2620ms
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstdlib>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 500000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
typedef long long LL;
const int N=5e5+5; int Cnt,fa[N],Enum,H[N],nxt[N],to[N],F[N],sz[N];
LL w[N];
char IN[MAXIN],*SS=IN,*TT=IN;
struct Node
{
int x,sz; LL val;
bool operator <(const Node &x)const
{
return val*x.sz>x.val*sz;
}
}; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
inline void AE(int u,int v)
{
to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum;
}
void DFS(int x)
{
static bool vis[N];
++Cnt, vis[x]=1;
for(int i=H[x]; i; i=nxt[i])
if(!vis[to[i]]) DFS(to[i]);
else {puts("-1"); exit(0);}
}
int Find(int x)
{
return x==F[x]?x:F[x]=Find(F[x]);
} int main()
{
// freopen("perm.in","r",stdin);
// freopen("perm.out","w",stdout); const int n=read();
for(int i=1; i<=n; ++i) AE(fa[i]=read(),i);
DFS(0);
if(Cnt<=n) return puts("-1"),0;
std::priority_queue<Node> q;
for(int i=0; i<=n; ++i) F[i]=i, sz[i]=1;//from 0
for(int i=1; i<=n; ++i) q.push((Node){i,1,w[i]=read()});
LL ans=0;
while(!q.empty())
{
Node tmp=q.top(); q.pop();
int x=Find(tmp.x);
if(tmp.sz!=sz[x]) continue;
int r=Find(fa[x]);
F[x]=r, ans+=tmp.val*sz[r], sz[r]+=sz[x], w[r]+=w[x];
if(r) q.push((Node){r,sz[r],w[r]});
}
printf("%lld\n",ans); return 0;
}

heap

//25792kb	2716ms
#include <cstdio>
#include <cctype>
#include <cstdlib>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 500000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
typedef long long LL;
const int N=5e5+5; int Cnt,fa[N],Enum,H[N],nxt[N],to[N],F[N],sz[N];
LL w[N];
char IN[MAXIN],*SS=IN,*TT=IN;
struct Node
{
int x,sz; LL val;
bool operator <(const Node &x)const
{
return val*x.sz>x.val*sz;
}
}q[N]; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
inline void AE(int u,int v)
{
to[++Enum]=v, nxt[Enum]=H[u], H[u]=Enum;
}
void DFS(int x)
{
static bool vis[N];
++Cnt, vis[x]=1;
for(int i=H[x]; i; i=nxt[i])
if(!vis[to[i]]) DFS(to[i]);
else {puts("-1"); exit(0);}
}
int Find(int x)
{
return x==F[x]?x:F[x]=Find(F[x]);
} int main()
{
// freopen("perm.in","r",stdin);
// freopen("perm.out","w",stdout); const int n=read();
for(int i=1; i<=n; ++i) AE(fa[i]=read(),i);
DFS(0);
if(Cnt<=n) return puts("-1"),0;
for(int i=0; i<=n; ++i) F[i]=i, sz[i]=1;//from 0
for(int i=1; i<=n; ++i) q[i]=(Node){i,1,w[i]=read()};
LL ans=0; int cnt=n;
std::make_heap(q+1,q+1+cnt);
while(cnt)
{
Node tmp=q[1]; std::pop_heap(q+1,q+1+cnt), --cnt;
int x=Find(tmp.x);
if(tmp.sz!=sz[x]) continue;
int r=Find(fa[x]);
F[x]=r, ans+=tmp.val*sz[r], sz[r]+=sz[x], w[r]+=w[x];
if(r) q[++cnt]=(Node){r,sz[r],w[r]}, std::push_heap(q+1,q+1+cnt);
}
printf("%lld\n",ans); return 0;
}

BZOJ.5289.[AHOI/HNOI2018]排列(贪心 heap)的更多相关文章

  1. BZOJ.5288.[AHOI/HNOI2018]游戏(思路 拓扑)

    BZOJ LOJ 洛谷 考虑如何预处理每个点能到的区间\([l,r]\). 对于\(i,i+1\)的一扇门,如果钥匙在\(i\)的右边,连边\(i\to i+1\),表示从\(i\)出发到不了\(i+ ...

  2. BZOJ.5290.[AHOI/HNOI2018]道路(树形DP)

    BZOJ LOJ 洛谷 老年退役选手,都写不出普及提高DP= = 在儿子那统计贡献,不是在父亲那统计啊!!!(这样的话不写这个提高DP写记忆化都能过= =) 然后就令\(f[x][a][b]\)表示在 ...

  3. BZOJ.5287.[AHOI HNOI2018]毒瘤(虚树 树形DP)

    BZOJ LOJ 洛谷 设\(f[i][0/1]\)表示到第\(i\)个点,不选/选这个点的方案数.对于一棵树,有:\[f[x][0]=\prod_{v\in son[x]}(f[v][0]+f[v] ...

  4. BZOJ.5285.[AHOI/HNOI2018]寻宝游戏(思路 按位计算 基数排序..)

    BZOJ LOJ 洛谷 话说vae去年的专辑就叫寻宝游戏诶 只有我去搜Mystery Hunt和infinite corridor了吗... 同样按位考虑,假设\(m=1\). 我们要在一堆\(01\ ...

  5. BZOJ.5286.[AHOI/HNOI2018]转盘(线段树)

    BZOJ LOJ 洛谷 如果从\(1\)开始,把每个时间\(t_i\)减去\(i\),答案取决于\(\max\{t_i-i\}\).记取得最大值的位置是\(p\),答案是\(t_p+1+n-1-p=\ ...

  6. 5289: [Hnoi2018]排列

    5289: [Hnoi2018]排列 链接 分析: 首先将题意转化一下:每个点向a[i]连一条边,构成了一个以0为根节点的树,要求选一个拓扑序,点x是拓扑序中的第i个,那么价值是i*w[x].让价值最 ...

  7. 【BZOJ5289】[HNOI2018]排列(贪心)

    [BZOJ5289][HNOI2018]排列(贪心) 题面 BZOJ 洛谷 题解 这个限制看起来不知道在干什么,其实就是找到所有排列\(p\)中,\(p_k=x\),那么\(k<j\),其中\( ...

  8. loj #2509. 「AHOI / HNOI2018」排列

    #2509. 「AHOI / HNOI2018」排列   题目描述 给定 nnn 个整数 a1,a2,…,an(0≤ai≤n),以及 nnn 个整数 w1,w2,…,wn.称 a1,a2,…,an 的 ...

  9. Loj #2495. 「AHOI / HNOI2018」转盘

    Loj #2495. 「AHOI / HNOI2018」转盘 题目描述 一次小 G 和小 H 原本准备去聚餐,但由于太麻烦了于是题面简化如下: 一个转盘上有摆成一圈的 \(n\) 个物品(编号 \(1 ...

随机推荐

  1. 高斯消元处理无解|多解情况 poj1830

    高斯消元结束后,若存在系数为0,常数不为0的行,则方程无解 若系数不为0的行有k个,则说明主元有k个,自由元有n-k个,方程多解 /* 给定n个开关的初始状态si,要求将其变成目标状态di 规定: 每 ...

  2. 20165314 2016-2017-2 《Java程序设计》第3周学习总结

    20165314 2016-2017-2 <Java程序设计>第3周学习总结 教材学习内容总结 类体包含成员变量和域变量 局部变量只在方法内有效 对象的创建以及对象对自己变量和方法通过用. ...

  3. 利用vue-cli创建新项目

    1.安装vue-cli npm i vue-cli --gd 2.初始化一个项目 vue init webpack test //test 是个项目名称并且配置相应的配置,(测试部份的可以选择no) ...

  4. MySQL5.7.20安装过程报错CMake Error at cmake/boost.cmake:81 (MESSAGE):

    MySQL在5.7版本及以后,都需要boots 库,所以需要先安装boots 步骤: 1.在/usr/local下创建 名为boots的目录 mkdir -p /usr/local/boots 2.进 ...

  5. spring cloud 使用ribbon简单处理客户端负载均衡

    假如我们的multiple服务的访问量剧增,用一个服务已经无法承载, 我们可以把Hello World服务做成一个集群. 很简单,我们只需要复制Hello world服务,同时将原来的端口8762修改 ...

  6. JAVA追加写入文本文件

    public void method1() { FileWriter fw = null; try { //如果文件存在,则追加内容:如果文件不存在,则创建文件 File f=new File(&qu ...

  7. java.lang.NoClassDefFoundError: javax/servlet/AsyncListener解决方案

    问题:spring3.2的架构在tomcat6.0中无法正常启动,抛出java.lang.NoClassDefFoundError: javax/servlet/AsyncListener错误 原因: ...

  8. STL容器之优先队列

    STL容器之优先队列 优先级队列,以前刷题的时候用的比较熟,现在竟然我只能记得它的关键字是priority_queue(太伤了).在一些定义了权重的地方这个数据结构是很有用的. 先回顾队列的定义:队列 ...

  9. Java开发环境笔记

    在配置环境变量中 设置Java_home: 一是为了方便引用,比如,jdk安装在c:\jdk16.0目录里,则设置java_home为该目录路径,那么以后要使用这个路径的时候,只需输入%java_ho ...

  10. ***小程序wx.getUserInfo不能弹出授权窗口后的解决方案

    微信更新api后,wx.getUserInfo在开发和体验版本都不能弹出授权窗口.微信文档说明: 注意:此接口有调整,使用该接口将不再出现授权弹窗,请使用 <button open-type=& ...