[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上的边,如果加入这条边,那么肯定会在这棵树上形成一 ...
随机推荐
- 【转】nodejs接收前端formData数据
很多时候需要利用formdata数据格式进行前后端交互. 前端代码可以是如下所示: <!DOCTYPE html> <html lang="en"> < ...
- android: Canvas的drawArc()方法的几个误区
绘制圆环很多时候会用到Canvas的drawArc方法, drawArc()方法的说明很简单: public void drawArc (RectF oval, float startAngle, f ...
- PAT 甲级 1055 The World's Richest (25 分)(简单题,要用printf和scanf,否则超时,string 的输入输出要注意)
1055 The World's Richest (25 分) Forbes magazine publishes every year its list of billionaires base ...
- 【转】hr的嘴,骗人的鬼
入职前,从上往下读,入职后,从下往上读. - 我们非常欢迎新鲜血液补充进我们的团队:- 如果条件太苛刻,我待不久的:- 我们公司绝对不可能这样:- 请问每个月的工作都能按时足额发放吗?- 这难道不是理 ...
- iOS摄像头和相册(转)
iOS摄像头和相册iOS 获取图片有三种方法1. 直接调用摄像头拍照 2. 从相册中选择 3. 从图库中选择 UIImagePickerController 是系统提供的用来获取图片和视频的接口: 用 ...
- 用Python做一个简单的小游戏
学习总是枯燥的,对于Python小白的我来讲,更是乏味的.为了提高学习的兴趣,今天我就来写一个小程序练练手. 数字猜谜游戏相信大家都不陌生,A给出最小值最大值,B写一个该范围内的数,A猜测写下的是多少 ...
- 高级UI-DrawerLayout侧滑
侧滑的方案有很多种,早期的开源SliddingMenu,以及后来的DrawerLayout以及NavigationView等都可实现侧滑效果,这里介绍的是DrawerLayout,下一节将介绍Navi ...
- 最新 房天下java校招面经 (含整理过的面试题大全)
从6月到10月,经过4个月努力和坚持,自己有幸拿到了网易雷火.京东.去哪儿.房天下等10家互联网公司的校招Offer,因为某些自身原因最终选择了房天下.6.7月主要是做系统复习.项目复盘.LeetCo ...
- linux中查看磁盘容量的常用操作
linux中查看磁盘容量常用操作 实验室有GPU集群,用户跑数据时候跑着跑着会出现集群挂掉的问题,原因就是,在跑数据时,用户上传文件,数据集,系统产生缓存等一系列操作,消耗了集群空间,师兄让我清理下服 ...
- 系统获取 IP 工具类
系统获取 IP 工具类 import java.net.Inet4Address; import java.net.InetAddress; import java.net.NetworkInterf ...