luogu P4437 [HNOI/AHOI2018]排列
问题本质是把\(a_i\)作为\(i\)的父亲,然后如果有环就不合法,否则每次要取数,要满足取之前他的父亲都被取过(父亲为0可以直接取),求最大价值
贪心想法显然是要把权值大的尽量放在后面,这等价于把权值小的尽量放在前面.所以如果当前最小的数没有父亲,显然直接取出来最优;如果有父亲,那么这个数应该在它的父亲被取之后马上取出来.这时我们把这两个点合并.之后重复此操作知道所有点被取完,就能得到答案
还有个问题是两个点合并后怎么取权值.两个点合并相当于两个序列合并,序列分别记为\(\{a_1,a_2...a_n\},\{b_1,b_2...b_m\}\),考虑什么时候\(\{a\}\)会放在\(\{b\}\)前面,\(\{a\}\)在前面的答案为\(ans_a+ans_b+n\sum_{j=1}^{m}b_j\),\(\{b\}\)在前面的答案为\(ans_a+ans_b+m\sum_{i=1}^{n}a_i\),\(\{a\}\)在前面当且仅当\(n\sum_{j=1}^{m}b_j\ge m\sum_{i=1}^{n}a_i\),等价于\(\frac{\sum a_i}{n}\le \frac{\sum b_j}{m}\),所以把权值设为里面点点权平均值即可.然后两个点\(a,b\)合并,会产生\(n\sum_{j=1}^{m}b_j\)的贡献,直接往答案里加即可
#include<bits/stdc++.h>
#define LL long long
#define uLL unsigned long long
#define db double
using namespace std;
const int N=5e5+10;
int rd()
{
	int x=0,w=1;char ch=0;
	while(ch<'0'||ch>'9'){if(ch=='-') w=-1;ch=getchar();}
	while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+(ch^48);ch=getchar();}
	return x*w;
}
struct node
{
	LL w,sz,i;
	bool operator < (const node &bb) const {return w*bb.sz!=bb.w*sz?w*bb.sz>bb.w*sz:i>bb.i;}
	bool operator == (const node &bb) const {return w==bb.w&&sz==bb.sz&&i==bb.i;}
}a[N];
bool ban[N];
struct HEAP
{
	priority_queue<node> q1;
	void mntn(){while(!q1.empty()&&(ban[q1.top().i]||!(q1.top()==a[q1.top().i]))) q1.pop();}
	void push(node x){q1.push(x);}
	void pop(){mntn();q1.pop();}
	node top(){mntn();return q1.top();}
}hp;
int n,fa[N],ff[N];
LL ans,sm;
int findf(int x){return ff[x]==x?x:ff[x]=findf(ff[x]);}
int main()
{
	n=rd();
	for(int i=1;i<=n;++i) ff[i]=i;
	for(int i=1;i<=n;++i)
	{
		fa[i]=rd();
		if(fa[i])
		{
			int x=findf(i),y=findf(fa[i]);
			if(x==y){puts("-1");return 0;}
			ff[y]=x;
		}
	}
	for(int i=1;i<=n;++i)
	{
		int w=rd();
		hp.push((a[i]=(node){w,1,i}));
		sm+=w;
	}
	ans=sm;
	int gg=0;
	for(int i=1;i<=n;++i) ff[i]=i;
	for(int i=1;i<=n;++i)
	{
		int x=hp.top().i;
		hp.pop();
		if(findf(fa[x]))
		{
			int xx=findf(fa[x]);
			ans+=a[xx].sz*a[x].w;
			a[xx].w+=a[x].w,a[xx].sz+=a[x].sz;
			hp.push(a[xx]);
			ff[x]=xx;
		}
		else ff[x]=0,sm-=a[x].w,ans+=a[x].sz*sm;
		ban[x]=1;
	}
	for(int i=1;i<=n;++i) ff[i]=findf(i);
	printf("%lld\n",ans);
	return 0;
}
												
											luogu P4437 [HNOI/AHOI2018]排列的更多相关文章
- 洛谷 P4437 [HNOI/AHOI2018]排列(贪心+堆,思维题)
		
题面传送门 开始 WA ycx 的遗产(bushi 首先可以将题目转化为图论模型:\(\forall i\) 连边 \(a_i\to i\),然后求图的一个拓扑序 \(b_1,b_2,\dots b_ ...
 - 【LG4437】[HNOI/AHOI2018]排列
		
[LG4437][HNOI/AHOI2018]排列 题面 洛谷 题解 题面里这个毒瘤的东西我们转化一下: 对于\(\forall k,j\),若\(p_k=a_{p_j}\),则\(k<j\). ...
 - 【题解】Luogu P4436 [HNOI/AHOI2018]游戏
		
原题传送门 \(n^2\)过百万在HNOI/AHOI2018中真的成功了qwqwq 先将没门分格的地方连起来,枚举每一个块,看向左向右最多能走多远,最坏复杂度\(O(n^2)\),但出题人竟然没卡(建 ...
 - 【洛谷 P4437】 [HNOI/AHOI2018]排列(贪心,堆)
		
题目链接 如果\(j<=k,a_{p[j]}!=p[k]\)可以理解为如果\(a_{p[j]}=p[k]\),那么\(k\)一定要放在\(j\)前面,也就是\(a_j\)在\(j\)前面. 于是 ...
 - BZOJ5289 & 洛谷4437:[HNOI/AHOI2018]排列——题解
		
https://www.lydsy.com/JudgeOnline/problem.php?id=5289 https://www.luogu.org/problemnew/show/P4437 考虑 ...
 - [HNOI/AHOI2018]排列 贪心
		
题面 题解: 把题面的限制换成中文: 如果排在第k位的下标 = 排在第j位的值 ,那么k < j 换一个描述方式: 一个值为x的数要排在第x个数后面. 再换一个描述方式: \(fa[i] = a ...
 - [HNOI/AHOI2018]排列
		
[Luogu4437] 如果\(a[i]=j\)则序列\(p[]\)中\(j\)必须排在\(i\)前面,如果\(j\)不在范围内则不管,求一个式子\(\sum_{i=1}^n iw_{p[i]}\)的 ...
 - 【题解】Luogu P4438 [HNOI/AHOI2018]道路
		
原题传送门 实际就是一道简单的树形dp 设f[u][i][j]表示从根结点到结点u经过i条未翻修公路,j条未翻修铁路的贡献最小值 边界条件:f[leaf][i][j]=(A+i)(B+j)C (题目上 ...
 - BZOJ5289 HNOI/AHOI2018排列(贪心+堆)
		
题面描述的相当绕,其实就是如果ai=j,重排后ai要在aj之后.同时每个ai有附属属性wi,要求最大化重排后的Σiwi. 容易发现这事实上构成一张图,即由j向i连边.由于每个点入度为1或0,该图是基环 ...
 
随机推荐
- 安装vncserver, vncviewer--远程桌面
			
1 问题如下 /etc/sysconfig/vncservers---配置文件作用去掉最后两行的注释 no route to host 是防火墙的原因---必须得研究好防火墙 本地可以vnc,本地 ...
 - Maven-指定要打包的文件
			
在项目 pom.xml 中指定 <build> <resources> <resource> <!--控制资源目录下要打包进去的文件,这里为全部打包--> ...
 - cygwin执行.py提示找不到模块,但已经安装模块的解决办法
			
. 在解决了cygwin中make命令不能使用的问题之后(https://www.cnblogs.com/zhenggege/p/10724122.html),make maskrcnn路径下的set ...
 - Vue常见的框架
			
1. Element:一套为开发者,设计师和产品经理准备的基于Vue 2.0的桌面端组件库 地址:https://element.eleme.cn/#/zh-CN 2.iview:主要服务于PC界面的 ...
 - [go]os.Open方法源码
			
file, err := os.Open("./buf.go") func Open(name string) (*File, error) { return OpenFile(n ...
 - 算法试题 - 找出最小 k 个数
			
题目 题目:输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 解析 思路1 这一题应用堆排序算法复杂度只有O(nlog k), ...
 - springboot2.0双数据源配置
			
题记:由于项目中不只是用一个数据库,所以记下以免忘记. 1.首先展示目录结构 2.pom配置文件 <?xml version="1.0" encoding="UTF ...
 - .net core 入门一
			
官网教程:https://docs.microsoft.com/zh-cn/aspnet/core/getting-started/?view=aspnetcore-3.0&tabs=wind ...
 - Markdown 介绍
			
Markdown 是目前互联网上最流行的写作语言,它使用一些简单的符号(* / ` > [] () #)来标记文本格式,其简洁的语法.优美的格式以及强大的软件支持深受广大网友的喜爱.维基百科上对 ...
 - nginx 配置方向代理出错 The character [_] is never valid in a domain name
			
nginx 配置方向代理出错 The character [_] is never valid in a domain name 下面是配置信息: 原因是使用的tomcat为8及以上的版本时upstr ...