[JZOJ5888]GCD生成树
[JZOJ5888]GCD生成树
题目链接
gugugu
分析
对于N很小的情况,暴力Prim即可
对于值域很小的情况,我的想法与solution不太一样,将值相同的缩成一个点,\(O(w^2)\)预处理出每个数对的\(gcd\)
然后设立两个点集,一个加入生成树集合中的,另一个没有,一开始将1放在1集合,然后遍历值域,将存在的数与1构成的数对的gcd都存在一个\(multiset\)中,然后选最大的.加入一个数后又重复以上操作.时间复杂度似乎是\(O(w^2 log w)\)
正解就比较妙了(还是因为我太菜了)
一个显然的性质就是我们要尽量地选\((w,k \times w)\)这样的数对
于是我们从大到小钦定这个\(w\),枚举它的倍数
但是发现有种情况就是\(k_1\)和\(k_2\)不互质,这样的话看起来选的数对就是错的了,因为它们gcd显然比\(w\)更大
然而我们可以使用并查集维护联通块来避免这种情况
对值域建立联通块,我们每将一对数\((k_1 \times w, k_2 \times w)\)算进答案就将\(k_1 \times w\)和\(k_2 \times w\)所在联通块合并
这样的话就算你\(k_1,k_2\)不互质,由于我们从大到小枚举这个\(w\),这种情况已经是被算过的,你只要每次判断两个数是否在一个联通块内即可去掉这种情况
计算答案就很简单了,你枚举\(w\)倍数时加入了\(k\)对数,那么答案就加上\(k \times w\)
但是还要注意我们把值域相同的缩成一个点,显然相同值的点之间连生成树边也是一种最优的情况
于是再对每一个出现的值\(w\),记录出现次数\(rnk[w]\),答案加上\(w \times (rnk[w]-1)\)
又跑进了rank1 hhh
代码
/*
code by RyeCatcher
*/
const int maxn=100005;
const int inf=0x7fffffff;
int n,a[maxn],mx=-1;
int fa[maxn],rnk[maxn];
bool vis[maxn];
int get(int x){return (fa[x]==x)?fa[x]:(fa[x]=get(fa[x]));}
ll ans=0;
int main(){
int x,y,p;
//FO(gcd);
read(n);
for(ri i=1;i<=n;i++){
read(a[i]);
if(vis[a[i]]){
rnk[a[i]]++;
}
else{
vis[a[i]]=1;
fa[a[i]]=a[i];
rnk[a[i]]=1;
}
mx=max(mx,a[i]);
}
int pre=0;
for(ri k=mx;k>=1;k--){
pre=0,y=0;
for(ri o=k;o<=mx;o+=k){
if(vis[o]){
if(!pre)pre=get(o);
else {
x=get(o);
if(pre!=x){
y++;
if(rnk[pre]>rnk[x])fa[x]=pre;
else fa[pre]=x,pre=x;
}
}
}
}
if(vis[k])ans+=1ll*(y+rnk[k]-1)*k;//别忘了乘以边权
else ans+=1ll*k*y;
}
printf("%lld\n",ans);
return 0;
}
[JZOJ5888]GCD生成树的更多相关文章
- [JZOJ 5888] [NOIP2018模拟9.29] GCD生成树 解题报告 (最大生成树+公约数)
题目链接: http://172.16.0.132/senior/#main/show/5888 题目: 题解: 思路是这样的:两个数的最大公约数一定不会比这两个数的任意一个数大.因此我们把权值相等的 ...
- NOIP前的刷题记录
因为这几天要加油,懒得每篇都来写题解了,就这里记录一下加上一句话题解好了 P4071 [SDOI2016]排列计数 组合数+错排 loj 6217 扑克牌 暴力背包 P2511 [HAOI2008 ...
- 2018.9.25 NOIP模拟赛
*注意:这套题目应版权方要求,不得公示题面. 从这里开始 Problem A XOR Problem B GCD Problem C SEG 表示十分怀疑出题人水平,C题数据和标程都是错的.有原题,差 ...
- Codeforces 990 调和级数路灯贪心暴力 DFS生成树两子树差调水 GCD树连通块暴力
A 水题 /*Huyyt*/ #include<bits/stdc++.h> #define mem(a,b) memset(a,b,sizeof(a)) using namespace ...
- gcd最大生成树模板
出处: ACM International Collegiate Programming Contest, Egyptian Collegiate Programming ContestArab Ac ...
- hdu5398 GCD Tree(lct)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud GCD Tree Time Limit: 5000/2500 MS (Java/O ...
- 生成树题目泛做(AD第二轮)
题目1: NOI2014 魔法森林 LCT维护MST.解题报告见LOFTER #include <cstdio> #include <iostream> #include &l ...
- UVa 10766 Organising the Organisation (生成树计数)
题意:给定一个公司的人数,然后还有一个boss,然后再给定一些人,他们不能成为直属上下级关系,问你有多少种安排方式(树). 析:就是一个生成树计数,由于有些人不能成为上下级关系,也就是说他们之间没有边 ...
- kuangbin带你飞 生成树专题 : 次小生成树; 最小树形图;生成树计数
第一个部分 前4题 次小生成树 算法:首先如果生成了最小生成树,那么这些树上的所有的边都进行标记.标记为树边. 接下来进行枚举,枚举任意一条不在MST上的边,如果加入这条边,那么肯定会在这棵树上形成一 ...
随机推荐
- hadoop:/bin/bash: /bin/java: No such file or directory
Stack trace: ExitCodeException exitCode=127 In HADOOP_HOME/libexec/hadoop-config.sh look for the if ...
- 第一章 Shiro简介——《跟我学Shiro》
转发地址:https://www.iteye.com/blog/jinnianshilongnian-2018936 目录贴:跟我学Shiro目录贴 1.1 简介 Apache Shiro是Java ...
- self-attention详解
编写你自己的 Keras 层 对于简单.无状态的自定义操作,你也许可以通过 layers.core.Lambda 层来实现.但是对于那些包含了可训练权重的自定义层,你应该自己实现这种层. 这是一个 K ...
- python:序列化与反序列化(json、pickle、shelve)
本节内容 前言 json模块 pickle模块 shelve模块 总结 一.前言 1. 现实需求 每种编程语言都有各自的数据类型,其中面向对象的编程语言还允许开发者自定义数据类型(如:自定义类),Py ...
- 【c# 学习笔记】析构函数
析构函数 用于在类销毁之前释放类实例所使用的托管和非托管资源.对应c#应用程序所创建的大多数对象,可以依靠.net Framework的垃圾回收站(GC) 来隐式地执行内存管理任务.但若创建封装了非托 ...
- linux SSH 登录
1: 远程连接linux server: ssh -p22 root@192.168.XXX.XXX 然后输入密码. 2: 免密码的方式: A: sudo -i B: ssh-keyge ...
- Npcap.资料
1.ZC:Npcap 是 WinPcap停更之后的继承者.我尝试了 Win7中发送raw tcp syn,它的代码和 使用WinPcap的基本一致. Developing software with ...
- 最新 腾讯java校招面试题(含整理过的面试题大全)
从6月到10月,经过4个月努力和坚持,自己有幸拿到了网易雷火.京东.去哪儿.腾讯等10家互联网公司的校招Offer,因为某些自身原因最终选择了腾讯.6.7月主要是做系统复习.项目复盘.LeetCode ...
- 使用Rabbit MQ消息队列
使用Rabbit MQ消息队列 综合概述 消息队列 消息队列就是一个消息的链表,可以把消息看作一个记录,具有特定的格式以及特定的优先级.对消息队列有写权限的进程可以向消息队列中按照一定的规则添加新消息 ...
- PLSQL设置文件夹排序并设置颜色