这题原题。。。

这题题面七绕八绕,有点麻烦,反正最后转化就是一棵树,每个点有一个值,要把所有点选完,要求选择一个点必须是它的父亲和祖先已经全部被选了,贡献是这个点的权值乘上它被选择的排名

如果一个点是它的父亲的所有儿子中权值最小的点,那么只要它的父亲选了,那接下来就肯定是选它。所以在序列中这个点的父亲是和它相邻的,可以直接合并了

然后看两个序列合并是如何合并的

例如长 \(m_1\)​ 的序列 \(a\) 和长 \(m_2\) 的序列 \(b\),和并后会放在整个序列的第 \(i\) 位置之后

如果 \(a\) 在 \(b\) 前面,贡献为\(\sum_{j=1}^{m_1}(i+j)w_{a_j}+\sum_{j=1}^{m_2}(i+j+m_1)w_{b_j}\)

如果 \(a\) 在 \(b\) 后面,贡献为\(\sum_{j=1}^{m_2}(i+j)w_{b_j}+\sum_{j=1}^{m_1}(i+j+m_2)w_{a_j}\)

然后我们推一推

\(\sum_{j=1}^{m_1}(i+j)w_{a_j}+\sum_{j=1}^{m_2}(i+j+m_1)w_{b_j}=\sum_{j=1}^{m_1}(i+j)w_{a_j}+\sum_{j=1}^{m_2}(i+j)w_{b_j}+m_1perm_b\)

\(\sum_{j=1}^{m_2}(i+j)w_{b_j}+\sum_{j=1}^{m_1}(i+j+m_2)w_{a_j}=\sum_{j=1}^{m_2}(i+j)w_{b_j}+\sum_{j=1}^{m_1}(i+j)w_{a_j}+m_2perm_a\)

作差 \(perm_{ab}-perm_{ba}=m_1perm_b-m_2perm_a\)

假如\(m_1perm_b-m_2perm_a>0\) 即 \(\frac{perm_a}{m_1}\lt\frac{perm_b}{m_2}\) ,那么 \(ba\) 比 \(ab\) 更优秀

所以就可贪心,按平均值贪心就好了

上一波pbds,因为它可以把堆和并查集放在一起做

  1. #include<bits/stdc++.h>
  2. #include<ext/pb_ds/assoc_container.hpp>
  3. #include<ext/pb_ds/priority_queue.hpp>
  4. #define ui unsigned int
  5. #define ll long long
  6. #define db double
  7. #define ld long double
  8. #define ull unsigned long long
  9. typedef std::pair<ld,int> pli;
  10. const int MAXN=500000+10;
  11. int n,e,beg[MAXN],to[MAXN<<1],nex[MAXN<<1],degree[MAXN],size[MAXN],treefa[MAXN],fa[MAXN],w[MAXN],s;
  12. ll res,val[MAXN];
  13. __gnu_pbds::priority_queue< pli,std::greater<pli> > q;
  14. __gnu_pbds::priority_queue< pli,std::greater<pli> >::point_iterator it[MAXN];
  15. template<typename T> inline void read(T &x)
  16. {
  17. T data=0,w=1;
  18. char ch=0;
  19. while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
  20. if(ch=='-')w=-1,ch=getchar();
  21. while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
  22. x=data*w;
  23. }
  24. template<typename T> inline void write(T x,char ch='\0')
  25. {
  26. if(x<0)putchar('-'),x=-x;
  27. if(x>9)write(x/10);
  28. putchar(x%10+'0');
  29. if(ch!='\0')putchar(ch);
  30. }
  31. template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
  32. template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
  33. template<typename T> inline T min(T x,T y){return x<y?x:y;}
  34. template<typename T> inline T max(T x,T y){return x>y?x:y;}
  35. inline void insert(int x,int y)
  36. {
  37. to[++e]=y;
  38. nex[e]=beg[x];
  39. beg[x]=e;
  40. }
  41. inline bool nosolution()
  42. {
  43. for(register int i=1;i<=n;++i)
  44. if(!degree[i])return false;
  45. puts("-1");
  46. return true;
  47. }
  48. inline int found(int x)
  49. {
  50. if(fa[x]!=x)fa[x]=found(fa[x]);
  51. return fa[x];
  52. }
  53. inline void dfs(int x,int f)
  54. {
  55. treefa[x]=f;
  56. for(register int i=beg[x];i;i=nex[i])
  57. if(to[i]!=f)dfs(to[i],x);
  58. }
  59. int main()
  60. {
  61. freopen("perm.in","r",stdin);
  62. freopen("perm.out","w",stdout);
  63. read(n);
  64. s=n+1;
  65. for(register int i=1,x;i<=n;++i)read(x),insert(x?(degree[i]++,x):s,i);
  66. for(register int i=1;i<=n;++i)read(w[i]);
  67. if(nosolution())return 0;
  68. dfs(s,0);
  69. fa[s]=s;val[s]=0;size[s]=1;
  70. for(register int i=1;i<=n;++i)
  71. {
  72. fa[i]=i;val[i]=w[i];size[i]=1;
  73. it[i]=q.push(std::make_pair((ld)val[i],i));
  74. }
  75. while(!q.empty())
  76. {
  77. int x=q.top().second,y=found(treefa[x]);
  78. q.pop();
  79. res+=1ll*val[x]*size[y];
  80. fa[x]=y;size[y]+=size[x];val[y]+=val[x];
  81. if(y==s)continue;
  82. q.modify(it[y],std::make_pair((ld)val[y]/size[y],y));
  83. }
  84. write(res,'\n');
  85. return 0;
  86. }

【比赛】HNOI2018 排列的更多相关文章

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

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

  2. 5289: [Hnoi2018]排列

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

  3. bzoj 5289: [Hnoi2018]排列

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

  4. [HNOI2018]排列

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

  5. [HNOI2018]排列[堆]

    题意 给定一棵树,每个点有点权,第 \(i\) 个点被删除的代价为 \(w_{p[i]}\times i\) ,问最小代价是多少. 分析 与国王游戏一题类似. 容易发现权值最小的点在其父亲选择后就会立 ...

  6. BZOJ5289: [Hnoi2018]排列

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

  7. loj2509 hnoi2018排列

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

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

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

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

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

随机推荐

  1. 跨域发送HTTP请求详解

    ------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥------------- 本篇博客讲述几种跨域发HTTP请求的几种方法,POST请求,GET请求 目录: 一,采用JsonP的方式(只能 ...

  2. android 签名相关

    查看keystorekeytool -list -v -keystore debug.keystoreapk签名不带别名 apksigner sign --ks debug.keystore test ...

  3. POJ-3273(二分)

    //题意:给出农夫在n天中每天的花费,要求把这n天分作m组, //每组的天数必然是连续的,要求分得各组的花费之和应该尽可能地小,最后输出各组花费之和中的最大值. //思路:看到各组最小和最大的,果断上 ...

  4. 深度学习论文笔记:Deep Residual Networks with Dynamically Weighted Wavelet Coefficients for Fault Diagnosis of Planetary Gearboxes

    这篇文章将深度学习算法应用于机械故障诊断,采用了“小波包分解+深度残差网络(ResNet)”的思路,将机械振动信号按照故障类型进行分类. 文章的核心创新点:复杂旋转机械系统的振动信号包含着很多不同频率 ...

  5. 微软职位内部推荐-Software Engineer II-Data Mining

    微软近期Open的职位: Are you looking for a big challenge? Do you know why Big Data is the next frontier for ...

  6. Focalprice李培亮:梦想让人在我店里排队

    [亿邦动力网讯]4月3日消息,外贸B2C平台Focalprice总裁李培亮日前亮相亿邦动力网联合河南省商务厅举办的“第九届中国中小企业电子商务大会暨2014中国(河南)跨境贸易电子商务峰会”,表达自己 ...

  7. Amazon及其亏本诱饵策略还能坚持多久?

    Amazon 刚刚公布了最新的财报,亏损 4100 万美元.这是这家电子商务巨头连续 5 个季度以来的第 3 次亏损.但是华尔街似乎却一片叫好声,当日 Amazon 的股价也涨了近 8 个点达 359 ...

  8. Requests库入门——应用实例-网络图片的爬取与保存(好看的小姐姐≧▽≦)

    在B站学习这一节的时候,弹幕最为激烈,不管大家是出于什么目的都想体验一下网络爬虫爬取图片的魅力,毕竟之前的实例实话说都是一些没有太大作用的信息. 好了,直接上代码: import requests i ...

  9. UML设计(团队作业6)

    决胜 Poker 一.团队成员 学号 姓名 211606392 郑俊瑜 (队长) 211606327 冉繁盛 211606323 刘世华 211606386 姚皓钰 211606358 陈卓楠 211 ...

  10. 20172321 2018-2019《Java软件结构与数据结构》第三周学习总结

    教材学习内容总结 第五章 5.1概述 队列是一种线性集合,其元素从一端加入,从另一端删除:队列的处理方式是先进先出(First in First out). 与栈的比较(LIFO) 栈是一端操作,先进 ...