Poj2054 color a tree && [HNOI/AHOI2018]排列
https://zybuluo.com/ysner/note/1120723
题面
原题
某省选强化题
大致意思是给你一颗树,选父亲后才能选儿子。
每个点对答案的贡献为你在第几次选这个点 × 该点权值
问取完所有点最小答案是多少。
- 对于\(60pts\) \(n\leq1000\)
- 对于\(100pts\) \(n\leq500000\)
解析
贪心没学好,_ _ _ _ _(自己yy)
答案要求的其实是个选取序列。。。
我们可以发现,要保证答案最优性,最大点取的时间越小越好。
又可以发现,只要最大点父亲已选,下一个点选最大点肯定最优。
这时我们确定了一部分顺序,可以想,为什么不能把已确定的、前后相邻选的两点合并起来呢?
现在问题就是如何在几个大点中作抉择了。
可以发现,此时如果大点\(a[i]/t[i]\)越大(\(a[i]\)是权值和,\(t[i]\)是时间和),就是性价比越高,先选越优。
对于\(n\leq1000\)
显然\(a[i]/t[i]\)看起来有点鬼
我们要用更有区分度的比较方式
如果\(a[i]/t[i]>a[j]/t[j]\)
那么\(a[i]*t[j]>a[j]*t[i]\)
于是每次按着贪心策略合并点,对答案的贡献为其父亲选完所用时间×点权值,再把新点与被合并点的儿子相连即可。
int main()
{
while(233)
{
memset(h,-1,sizeof(h));ans=0;cnt=0;
n=gi();r=gi();
if(!n&&!r) break;t[0]=1;
fp(i,1,n) a[i]=gi(),t[i]=1;
fp(i,1,n-1)
{
re int u=gi(),v=gi();
add(u,v);f[v]=u;
}
f[r]=0;
re int res=n;
while(res--)
{
re int mx=1;
fp(i,2,n)
if(a[i]*t[mx]>a[mx]*t[i]) mx=i;
ans+=a[mx]*t[f[mx]];
a[f[mx]]+=a[mx];t[f[mx]]+=t[mx];
for(re int i=h[mx];i+1;i=e[i].nxt)
{
re int v=e[i].to;
f[v]=f[mx];
add(f[mx],v);
}
a[mx]=-1;
}
printf("%lld\n",ans);
}
return 0;
}
Update:发现一个很蛋疼的问题:用并查集就不用新建边了。
对于\(n\leq500000\)
裸贪心是\(O(n^2)\)的。
我们需要一个能马上提供最大点的堆来降低复杂度。
struct node{int u,sz;ll w;bool operator < (const node &o) const {return w*o.sz>o.w*sz;}};
priority_queue<node> Q;
il void dfs(re int u)
{
vis[u]=1;++tim;
for(re int i=h[u];i+1;i=e[i].nxt)
{
re int v=e[i].to;
if(vis[v]) {puts("-1");exit(0);}
dfs(v);
}
}
il int find(re int x){return x==F[x]?x:F[x]=find(F[x]);}
int main()
{
memset(h,-1,sizeof(h));ans=0;
n=gi();
fp(i,1,n) f[i]=gi(),add(f[i],i);
dfs(0);if(tim<=n) return puts("-1"),0;
fp(i,0,n) F[i]=i,sz[i]=1;
fp(i,1,n) w[i]=gi(),Q.push((node){i,1,w[i]});
re int u,p;
while(!Q.empty())
{
node s=Q.top();Q.pop();u=find(s.u);
if(sz[u]^s.sz) continue;//扔掉不符合当前情况的决策
F[u]=p=find(f[u]);
ans+=w[u]*sz[p];
w[p]+=w[u];sz[p]+=sz[u];
if(p) Q.push((node){p,sz[p],w[p]});
}
printf("%lld\n",ans);
return 0;
}
Poj2054 color a tree && [HNOI/AHOI2018]排列的更多相关文章
- 【LG4437】[HNOI/AHOI2018]排列
[LG4437][HNOI/AHOI2018]排列 题面 洛谷 题解 题面里这个毒瘤的东西我们转化一下: 对于\(\forall k,j\),若\(p_k=a_{p_j}\),则\(k<j\). ...
- [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]}\)的 ...
- 洛谷 P4437 [HNOI/AHOI2018]排列(贪心+堆,思维题)
题面传送门 开始 WA ycx 的遗产(bushi 首先可以将题目转化为图论模型:\(\forall i\) 连边 \(a_i\to i\),然后求图的一个拓扑序 \(b_1,b_2,\dots b_ ...
- [POJ2054]Color a Tree (并查集+贪心)
POJ终于修好啦 题意 和UVA1205是同一题,在洛谷上是紫题 有一棵树,需要给其所有节点染色,每个点染色所需的时间是一样的都是11.给每个点染色,还有一个开销“当前时间×ci×ci”,cici是每 ...
- poj2054 Color a Tree
神题.这题是巨毒瘤... 自己写真可谓是: 排空驭气奔如电,上天入地求之遍 上穷碧落下黄泉,两处茫茫皆不见 由于我们知道:不是树形时,不停选值最大的节点可以得到最小代价. 那么我们就能想出一个错误的贪 ...
- BZOJ5289 HNOI/AHOI2018排列(贪心+堆)
题面描述的相当绕,其实就是如果ai=j,重排后ai要在aj之后.同时每个ai有附属属性wi,要求最大化重排后的Σiwi. 容易发现这事实上构成一张图,即由j向i连边.由于每个点入度为1或0,该图是基环 ...
- 【洛谷 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 考虑 ...
随机推荐
- 梦想CAD控件图块COM接口知识点
梦想CAD控件图块COM接口知识点 图块是将多个实体组合成一个整体,并给这个整体命名保存,在以后的图形编辑中图块就被视为一个实体.一个图块包括可见的实体如线.圆.圆弧以及可见或不可见的属性数据.图块的 ...
- ThinkPHP---案例2--职员管理功能
[一]准备工作 (1)创建菜单,修改跳转路径 <li> <a href="javascript:;" class="workerManage" ...
- 01JavaScript使用
JavaScript使用 1.内容写入 HTML <P onmouseover="alert('欢迎您学习JavaScript!')">鼠标移过来</P> ...
- nginx+tomcat+memcache
tomcat1和tomcat2都需要安装以下配置[root@tomcat-1 ~]# yum -y install gcc openssl-devel pcre-devel zlib-devel[ro ...
- 【6.24校内test】T1 江城唱晚
[题目背景] 墙角那株海棠,是你种下的思念. 生死不能忘,高烛照容颜. 一曲江城唱晚,重忆当年坐灯前, 青衫中绣着你留下的线. ——银临<江城唱晚> [问题描述] 扶苏是个喜欢一边听古风歌 ...
- spring boot+mybatis+mysql增删改查分页
server: port: servlet: context-path: /springBootMybatis spring: datasource: name: test url: jdbc:mys ...
- C++ string使用
在c语言里,我们使用一个字符串时,是通过字符数组或者字符指针的方式来进行使用,在C++里,标准模板库已经给我们提供了string类型(string是以类的方式提供给我们使用). 定义和初始化strin ...
- 洛谷—— P1450 [HAOI2008]硬币购物
P1450 [HAOI2008]硬币购物 硬币购物一共有$4$种硬币.面值分别为$c1,c2,c3,c4$.某人去商店买东西,去了$tot$次.每次带$di$枚$ci$硬币,买$si$的价值的东西.请 ...
- UVA - 10048 Audiophobia(Floyd求路径上最大值的最小)
题目&分析: 思路: Floyd变形(见上述紫书分析),根据题目要求对应的改变判断条件来解题. 代码: #include <bits/stdc++.h> #define inf 0 ...
- Turtle-可视化界面画圣诞树
圣诞节(Christmas)又称耶诞节.耶稣诞辰,译名为“基督弥撒”,是西方传统节日,起源于基督教,在每年公历12月25日.弥撒是教会的一种礼拜仪式.圣诞节是一个宗教节,因为把它当作耶稣的诞辰来庆祝, ...