我对贪心的理解:https://www.cnblogs.com/AKMer/p/9776293.html

题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=5289

首先我们转化题意:对于任意一个\(w_i\),只能在\(w_{a_i}\)被取之后才能被取走。存在某一取法\(p\)使得\(\sum\limits_{i=1}^{i=n}i*w_{p_i}\)取值最大,求最大值。

对于每一个\(i\)与\(a_i\)的约束,我们可以建一条边来描述。那么这题就跟Color a Tree​一模一样的,不过求的是最大值罢了,那我们每次选平均值最小的与父亲合并就行。如果不能构成树就无解,构成树或者森林就有解。我们以\(0\)为虚根,那么就必然是一棵树了。

如果不存在某个\(a_i\)等于\(0\),或者存在\(a_i=i\)那么就肯定无解,否则必然有解。

然而知道这些你还是\(A\)不了这道题。若你问我而出此言,且听我细细道来:

\(1\)、平均值不开\(long\) \(double\)过不了;

\(2\)、平均值不开\(long\) \(double\)过不了;

\(3\)、平均值不开\(long\) \(double\)过不了;

重要的事情说三遍。

  1. 某出题人欺我老无力,
  2. 忍能偷偷卡精度。
  3. 公然卡我的double
  4. Bug连天改不出,
  5. 归来伏桌自叹息。

取最小值用堆维护一下,合并维护父亲信息用并查集就可以了。

时间复杂度:\(O(nlogn)\)

空间复杂度:\(O(n)\)

代码如下:

  1. #include <queue>
  2. #include <cstdio>
  3. #include <algorithm>
  4. using namespace std;
  5. #define ll long long
  6. #define ld long double
  7. const int maxn=5e5+5;
  8. int n,tot;
  9. ll w[maxn],ans;
  10. bool bo1,bo2,vis[maxn];
  11. int a[maxn],fa[maxn],num[maxn],father[maxn];
  12. int read() {
  13. int x=0,f=1;char ch=getchar();
  14. for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
  15. for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
  16. return x*f;
  17. }
  18. struct node {
  19. int id;
  20. ld ave;
  21. bool operator<(const node &a)const {
  22. return ave>a.ave;//初始是大根堆,所以要反过来
  23. }
  24. };
  25. priority_queue<node> T;
  26. int find(int x) {
  27. if(fa[x]==x)return x;
  28. return fa[x]=find(fa[x]);
  29. }
  30. void work() {
  31. for(int i=1;i<=n;i++)
  32. T.push((node){i,(ld)w[i]});
  33. while(!T.empty()){
  34. node N=T.top();int u=N.id;T.pop();
  35. if(vis[u])continue;vis[u]=1;//由于每次都是将最小值合并,所以越合并平均值也就会越小,那么我们只需要知道这个node是否是已经合并过的node就行了。对于任意一个没有合并但是被多次插入堆中的node,最新的一次插入肯定是平均值最小的那一次,所以每次取最小值肯定就是最新更新的数据。
  36. int FA=find(father[u]);
  37. ans+=w[u]*num[FA];fa[u]=FA;
  38. w[FA]+=w[u];num[FA]+=num[u];
  39. if(FA)T.push((node){FA,(ld)w[FA]/num[FA]});//具体请见color a tree
  40. }
  41. }
  42. int main() {
  43. n=read();num[0]=1;
  44. for(int i=1;i<=n;i++) {
  45. a[i]=read();num[i]=1;
  46. fa[i]=i;father[i]=a[i];//father记树上父亲节点,fa记录并查集祖先
  47. if(!a[i])bo1=1;
  48. if(a[i]==i)bo2=1;
  49. }
  50. if(!bo1||bo2) {puts("-1");return 0;}//判无解
  51. for(int i=1;i<=n;i++)
  52. w[i]=read();
  53. work();printf("%lld\n",ans);
  54. return 0;
  55. }

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. 【BZOJ5289】[HNOI2018]排列(贪心)

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

  4. 5289: [Hnoi2018]排列

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

  5. bzoj 5289: [Hnoi2018]排列

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

  6. [HNOI2018]排列

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

  7. [HNOI2018]排列[堆]

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

  8. loj2509 hnoi2018排列

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

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

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

随机推荐

  1. 星球大战starwar(并查集)

    1015: [JSOI2008]星球大战starwar Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 5253  Solved: 2395[Submit ...

  2. transport connector和network connector

    1 什么是transport connector 用于配置activemq服务器端和客户端之间的通信方式. 2 什么是network connector 用于配置activemq服务器之间的通信方式, ...

  3. 测试站如何最快获取正式站的最新数据: ln -s

    针对静态数据, 比如图片/js等文件, 测试站如何获取最新的呢? ln -s /alidata/www/mysite/uploads /alidata/www/mysite_test/uploads ...

  4. Webpack探索【3】--- loader详解

    本文主要说明Webpack的loader相关内容.

  5. PHP网站在Linux服务器上面的安全配置

    本文详细总结了PHP网站在Linux服务器上面的安全配置,包含PHP安全.mysql数据库安全.web服务器安全.木马查杀和防范等,很好很强大很安全. PHP安全配置 1. 确保运行php的用户为一般 ...

  6. java面向对象入门之方法参数的传递

    /* Name : Power by :Stuart Date:2015.4.25 */ class PassOn{ //创建show方法,把i传入,输出i+1的结果 public void show ...

  7. sonarQube使用maven进行检查

    1.在maven的中找到setting配置文件.在setting.xml中增加sonarqube配置.如下: <profiles> <profile> <id>so ...

  8. mysql表数据压缩

    记得一次面试中,面试官问我是否知道表的压缩,这个时候我才知道mysql有个表压缩这么个功能,今天试用下看看表的压缩率怎么样. 这里分两个部分说明,第一部分:官方文档说明:第二部分:具体实例测试. [第 ...

  9. etcd -> Highly-avaliable key value store for shared configuration and service discovery

    The name "etcd" originated from two ideas, the unix "/etc" folder and "d&qu ...

  10. 遇到“拒绝了对对象的 EXECUTE 权限”和“无法作为数据库主体执行,因为主体 "dbo" 不存在、无法模拟这种类型的主体,或您没有所需的权限”的问题

    在将数据库从sqlserver2000迁移到2005后,原有的用户名TDS在执行存储过程是报错:“拒绝了对对象的 EXECUTE 权限”. 如网上所说,在使用的数据库的属性页->权限中给TDS添 ...