[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生成树的更多相关文章

  1. [JZOJ 5888] [NOIP2018模拟9.29] GCD生成树 解题报告 (最大生成树+公约数)

    题目链接: http://172.16.0.132/senior/#main/show/5888 题目: 题解: 思路是这样的:两个数的最大公约数一定不会比这两个数的任意一个数大.因此我们把权值相等的 ...

  2. NOIP前的刷题记录

    因为这几天要加油,懒得每篇都来写题解了,就这里记录一下加上一句话题解好了 P4071 [SDOI2016]排列计数   组合数+错排 loj 6217 扑克牌 暴力背包 P2511 [HAOI2008 ...

  3. 2018.9.25 NOIP模拟赛

    *注意:这套题目应版权方要求,不得公示题面. 从这里开始 Problem A XOR Problem B GCD Problem C SEG 表示十分怀疑出题人水平,C题数据和标程都是错的.有原题,差 ...

  4. Codeforces 990 调和级数路灯贪心暴力 DFS生成树两子树差调水 GCD树连通块暴力

    A 水题 /*Huyyt*/ #include<bits/stdc++.h> #define mem(a,b) memset(a,b,sizeof(a)) using namespace ...

  5. gcd最大生成树模板

    出处: ACM International Collegiate Programming Contest, Egyptian Collegiate Programming ContestArab Ac ...

  6. hdu5398 GCD Tree(lct)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud GCD Tree Time Limit: 5000/2500 MS (Java/O ...

  7. 生成树题目泛做(AD第二轮)

    题目1: NOI2014 魔法森林 LCT维护MST.解题报告见LOFTER #include <cstdio> #include <iostream> #include &l ...

  8. UVa 10766 Organising the Organisation (生成树计数)

    题意:给定一个公司的人数,然后还有一个boss,然后再给定一些人,他们不能成为直属上下级关系,问你有多少种安排方式(树). 析:就是一个生成树计数,由于有些人不能成为上下级关系,也就是说他们之间没有边 ...

  9. kuangbin带你飞 生成树专题 : 次小生成树; 最小树形图;生成树计数

    第一个部分 前4题 次小生成树 算法:首先如果生成了最小生成树,那么这些树上的所有的边都进行标记.标记为树边. 接下来进行枚举,枚举任意一条不在MST上的边,如果加入这条边,那么肯定会在这棵树上形成一 ...

随机推荐

  1. 【转载】 迁移学习简介(tranfer learning)

    原文地址: https://blog.csdn.net/qq_33414271/article/details/78756366 土豆洋芋山药蛋 --------------------------- ...

  2. Facebook libra白皮书

    https://libra.org/en-US/white-paper/ An Introduction to Libra Libra的使命是建立一个简单的全球货币和金融基础设施,为数十亿人服务.该文 ...

  3. 001-java 设计模式概述

    一.概述 思维导图 GoF(“四人帮”,又称Gang of Four,即Erich Gamma, Richard Helm, Ralph Johnson & John Vlissides) 1 ...

  4. Django之model详解

    Django中的页面管理后台 Djano中自带admin后台管理模块,可以通过web页面去管理,有点想php-admin,使用步骤: 在项目中models.py 中创建数据库表 class useri ...

  5. PHP上传文件 Error 6解决方法 (转)

    按:我采用phpstudy2016,编辑php.ini ,“upload_tmp_dir没放开,直接放开,并指向 /tmp 就OK 上传文件,$_FILES["file"][&qu ...

  6. 使用redis做为MySQL的缓存-C语言编写UDF

    介绍 在实际项目中,MySQL数据库服务器有时会位于另外一台主机,需要通过网络来访问数据库:即使应用程序与MySQL数据库在同一个主机中,访问MySQL也涉及到磁盘IO操作(MySQL也有一些数据预读 ...

  7. application.properties在Spring Boot项目中的位置

    application.properties可以放在如下位置: 当前目录的 "/config"的子目录下 当前目录下 classpath根目录的"/config" ...

  8. 解决kalilinux:“下列签名无效: KEYEXPIRED 1425567400"

    Kali linux由于太长时间未更新,而出现GPG错误 KEYEXPIRED 1425567400.经检查源未出现问题可以解析,deb也不冲突,就是密钥过期了. 解决方式 使用一条命令,添加新的密钥 ...

  9. react如何在网页上编辑并运行代码?

    最近想做个能在网站,能在网页上运行代码,并且保存这个组件,看了一下element-react的组件和官方的实例,发现都是可以编辑运行的,因为之前没有这方面的经验,所以看下各位大佬能不能给点意见

  10. 【VS开发】Windows平台下Makefile学习笔记

    作者:朱金灿 来源:http://blog.csdn.net/clever101 决心学习Makefile,一方面是为了解决编译开源代码时需要跨编译平台的问题(发现一些开源代码已经在使用VS2010开 ...