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

题面

BZOJ

洛谷

题解

这个限制看起来不知道在干什么,其实就是找到所有排列\(p\)中,\(p_k=x\),那么\(k<j\),其中\(a[p_j]=x\)。

也就是对于\(a\)数组的每个数\(a[i]\),它必须放在所有\(a[x]=i\)的前面。

那么对于\(i\)向所有满足\(a[x]=i\)的位置\(x\)连边,表示\(i\)必须放在这些数前面。

如果成环必定无解,如果无环则图是森林。

现在考虑每次从度数为\(0\)的点中选一个出来放在序列后面。

考虑这样一个问题,全局的最小值什么时候被选。

如果最小值入度为\(0\),显然直接被选。否则当它的父亲被选,那么它一定直接被选。

所以可以把最小值和其父亲合并在一起。

那么重复这个操作考虑每个联通块和他的父亲合并。

那么考虑两个块被选择的顺序关系,假设两个块\(a,b\),权值和为\(s_a,s_b\),点数为\(d_a,d_b\),那么:

\(W_{ab}=W_a+W_b+d_a*s_b\),\(W_{ba}=W_b+W_a+d_b*s_a\)。

不难发现如果\(a\)要放在\(b\)前面的话,就要满足\(d_a*s_b>d_b*s_a\)。即先选平均权值较小的块。

那么每次就选出这个块,然后把它和它的父亲合并在一起就好了,产生的贡献是\(d_a*s_b\)。

那么用\(set\)维护这个过程就做完了。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<set>
using namespace std;
#define ll long long
#define MAX 500500
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
struct Line{int v,next;}e[MAX<<1];
int h[MAX],cnt=1,tot;
inline void Add(int u,int v){e[cnt]=(Line){v,h[u]};h[u]=cnt++;}
int n,a[MAX],w[MAX],d[MAX];ll ans,W[MAX];
struct data{ll s;int d,r;};set<data> Q;
bool operator<(data a,data b){ll s1=1ll*a.s*b.d,s2=1ll*b.s*a.d;return s1==s2?a.r<b.r:s1<s2;}
int f[MAX];int getf(int x){return x==f[x]?x:f[x]=getf(f[x]);}
bool vis[MAX];int sz[MAX];
void dfs(int u,int ff)
{
vis[u]=true;sz[u]=1;
for(int i=h[u];i;i=e[i].next)
{
int v=e[i].v;if(v==ff)continue;
if(vis[v])puts("-1"),exit(0);
dfs(v,u);sz[u]+=sz[v];
}
}
data Get(int x){return (data){W[x],d[x],x};}
int main()
{
n=read();int pos=0;
for(int i=1;i<=n;++i)Add(a[i]=read(),i),f[i]=i;
for(int i=1;i<=n;++i)w[i]=read();
dfs(0,-1);if(sz[0]!=n+1){puts("-1");return 0;}
Q.insert((data){W[0]=1e18,d[0]=1,0});
for(int i=1;i<=n;++i)Q.insert((data){W[i]=w[i],d[i]=1,i});
while(!Q.empty())
{
data u=*Q.begin();Q.erase(u);if(!u.r)break;
int F=getf(a[u.r]);data v=Get(F);
Q.erase(v);ans+=u.s*v.d;
W[F]+=u.s;d[F]+=u.d;f[getf(u.r)]=F;
Q.insert(Get(F));
}
printf("%lld\n",ans);
return 0;
}

【BZOJ5289】[HNOI2018]排列(贪心)的更多相关文章

  1. BZOJ5289: [Hnoi2018]排列

    传送门 第一步转化,令 \(q[p[i]]=i\),那么题目变成: 有一些 \(q[a[i]]<q[i]\) 的限制,\(q\) 必须为排列,求 \(max(\sum_{i=1}^{n}w[i] ...

  2. [BZOJ5289][HNOI2018]排列(拓扑排序+pb_ds)

    首先确定将所有a[i]向i连边之后会形成一张图,图上每条有向边i->j表示i要在j之前选. 图上的每个拓扑序都对应一种方案(如果有环显然无解),经过一系列推导可以发现贪心策略与合并的块的大小和w ...

  3. BZOJ.5289.[AHOI/HNOI2018]排列(贪心 heap)

    BZOJ LOJ 洛谷 \(Kelin\)写的挺清楚的... 要求如果\(a_{p_j}=p_k\),\(k\lt j\),可以理解为\(k\)要在\(j\)之前选. 那么对于给定的\(a_j=k\) ...

  4. 5289: [Hnoi2018]排列

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

  5. BZOJ5289:[HNOI2018]排列

    我对贪心的理解:https://www.cnblogs.com/AKMer/p/9776293.html 题目传送门:https://www.lydsy.com/JudgeOnline/problem ...

  6. bzoj 5289: [Hnoi2018]排列

    Description Solution 首先注意到实际上约束关系构成了一棵树 考虑这个排列 \(p\),编号为 \(a[i]\) 的出现了,\(i\) 才可以出现 那么如果连边 \((a[i],i) ...

  7. [HNOI2018]排列

    Description: 给定 \(n\) 个整数 \(a_1, a_2, \dots, a_n, 0 \le a_i \le n\),以及 \(n\) 个整数 \(w_1, w_2, \dots, ...

  8. [HNOI/AHOI2018]排列 贪心

    题面 题解: 把题面的限制换成中文: 如果排在第k位的下标 = 排在第j位的值 ,那么k < j 换一个描述方式: 一个值为x的数要排在第x个数后面. 再换一个描述方式: \(fa[i] = a ...

  9. loj2509 hnoi2018排列

    题意:对于a数组,求它的一个合法排列的最大权值.合法排列:对于任意j,k,如果a[p[j]]=p[k],那么k<j. 权值:sigma(a[p[i]]*i).n<=50W. 标程: #in ...

随机推荐

  1. 【学习总结】GirlsInAI ML-diary day-6-String字符串

    [学习总结]GirlsInAI ML-diary 总 原博github链接-day6 认识字符串 字符串的性质 字符串的玩法 1-字符串就是字符的序列 序列,代表字符串是有顺序的!这里很重要. 比如我 ...

  2. 【问题解决方案】从 Anaconda Prompt 或 Jupyter Notebook 终端进入Python后重新退出到命令状态

    从 Anaconda Prompt 或 Jupyter Notebook 终端进入Python后重新退出到命令状态 退出Python:exit() 或者 Ctrl+z 例子一枚 默认打开的是3.7,需 ...

  3. Python爬虫——用BeautifulSoup、python-docx爬取廖雪峰大大的教程为word文档

    版权声明:本文为博主原创文章,欢迎转载,并请注明出处.联系方式:460356155@qq.com 廖雪峰大大贡献的教程写的不错,写了个爬虫把教程保存为word文件,供大家方便下载学习:http://p ...

  4. Redis教程(Linux)

    这里汇总了从简单的安装到较为复杂的配置,由浅入深的学习redis... 一 , 安装 1) redis扩展安装 从官网上下载扩展压缩包 wget http://pecl.php.net/get/red ...

  5. C#设计模式之4:装饰者模式

    装饰者模式 背景是有一家星巴兹咖啡店,由于客源充足,所以决定重新设计他们的收费系统,以前的收费系统中只定义了一个表示饮料的Beverage的基类,它里面定义了一个Cost的方法用来计算饮料的花费,但是 ...

  6. [转帖]NUMA

    作者:ibless 来源:CSDN 原文:https://blog.csdn.net/ibless/article/details/80114009 其实 很早之前对这一块有了解 比较多的的是 CCN ...

  7. [服务器]Gartner:2018年第四季度全球服务器收入增长17.8% 出货量增长8.5%

    Gartner:2018年第四季度全球服务器收入增长17.8% 出货量增长8.5% Gartner 是不是也是花钱买榜的主啊.. 简单看了一下 浪潮2018Q4的营收18亿刀 (季度营收110亿人民币 ...

  8. spring的xml配置里,最好不要配置xsd的版本名称

    <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.sp ...

  9. JS --- 如何获取一个对象的类型

    可以清楚的看到  拿到数字 字符串 对象 函数 数组 通过.slice(8,-1) 可以拿到类型的名称 ,可以做你想要的操作 Object.prototype.toString.call(222) & ...

  10. 动态渲染页面爬取(Python 网络爬虫) ---Selenium的使用

    Selenium 的使用 Selenium 是一个自动化测试工具,利用它可以驱动浏览器执行特定的动作,如点击.下拉等操作,同时还可以获取浏览器当前呈现的页面的源代码,做到可见即可爬.对于一些JavaS ...