正题

题目连接:https://www.luogu.com.cn/problem/P7854


题目大意

给出\(n\)数字的一个序列\(a\)。

现在要求构造一棵树,使得对于任意的\((x,y)\)都有

\[gcd(a_x,a_y)=a_{lca(x,y)}
\]

\(1\leq n\leq 10^5,1\leq a_i\leq 10^6\)


解题思路

考虑对于一个数字\(a_x\),我们枚举它的存在于\(a\)序列中所有约数\(a_d\),考虑对于这些\(a_d\)如果它们之间不存在祖先关系那么显然无解,否则我们就选择深度最大的那个节点连接。

当然枚举约数太麻烦所以我们直接枚举每个数的倍数。

然后这样的话发现其实是有问题的,因为我们只保证了\(a_{lca(x,y)}|gcd(a_x,a_y)\)。

但是有解时这样构造肯定是正确的,所以只需要考虑如何判断这种情况的无解即可。

发现如果对于每一对\((x,y)\)都存在\(a_{i}=gcd(a_x,a_y)\)那么就可以用上面那种情况构造。

所以我们只需要求出每个数字作为\(gcd(a_x,a_y)\)出现的次数就好了。

记\(m\)为\(max\{a_i\}\),那么时间复杂度就是\(O(n+m\log m)\)


解题思路

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
const int N=1e6+10,L=1e6;
int n,a[N],p[N],fa[N],r[N],c[N];
long long v[N];
bool cmp(int x,int y)
{return a[x]<a[y];}
int main()
{
scanf("%d",&n);int d=0;
for(int i=1;i<=n;i++)
scanf("%d",&a[i]),d=__gcd(d,a[i]);
for(int i=1;i<=n;i++)
a[i]=a[i]/d,p[i]=i,c[a[i]]++;
sort(p+1,p+1+n,cmp);
int z=1;
if(!c[1])return puts("-1")&0;
for(int i=1;i<=L;i++){
if(!c[i])continue;
while(z<=n&&a[p[z]]<=i){
fa[p[z]]=r[i];
r[i]=p[z];z++;
}
for(int j=2*i;j<=L;j+=i){
if(!c[j])continue;
if(!r[j])r[j]=r[i];
else{
if(i%a[r[j]])return puts("-1")&0;
r[j]=r[i];
}
}
}
for(int i=1;i<=L;i++){
for(int j=i;j<=L;j+=i)
v[i]+=c[j];
v[i]=v[i]*v[i];
}
for(int i=L;i>=1;i--)
for(int j=i+i;j<=L;j+=i)
v[i]-=v[j];
for(int i=1;i<=L;i++)
if(v[i]&&!c[i])return puts("-1")&0;
for(int i=1;i<=n;i++)
printf("%d ",fa[i]);
return 0;
}

P7854-「EZEC-9」GCD Tree【构造】的更多相关文章

  1. 「算法笔记」Link-Cut Tree

    一.简介 Link-Cut Tree (简称 LCT) 是一种用来维护动态森林连通性的数据结构,适用于动态树问题. 类比树剖,树剖是通过静态地把一棵树剖成若干条链然后用一种支持区间操作的数据结构维护, ...

  2. Solution -「AT 3913」XOR Tree

    \(\mathcal{Description}\)   Link.   给定一棵树,边 \((u,v)\) 有边权 \(w(u,v)\).每次操作可以使一条简单路径上的边权异或任意非负整数.求最少的操 ...

  3. Solution -「CF 1060F」Shrinking Tree

    \(\mathcal{Description}\)   Link.   给定一棵 \(n\) 个点的树,反复随机选取一条边,合并其两端两点,新点编号在两端两点等概率选取.问每个点留到最后的概率.    ...

  4. 洛谷比赛 「EZEC」 Round 4

    洛谷比赛 「EZEC」 Round 4 T1 zrmpaul Loves Array 题目描述 小 Z 有一个下标从 \(1\) 开始并且长度为 \(n\) 的序列,初始时下标为 \(i\) 位置的数 ...

  5. 【题解】#6622. 「THUPC 2019」找树 / findtree(Matrix Tree+FWT)

    [题解]#6622. 「THUPC 2019」找树 / findtree(Matrix Tree+FWT) 之前做这道题不理解,有一点走火入魔了,甚至想要一本近世代数来看,然后通过人类智慧思考后发现, ...

  6. 「学习笔记」Treap

    「学习笔记」Treap 前言 什么是 Treap ? 二叉搜索树 (Binary Search Tree/Binary Sort Tree/BST) 基础定义 查找元素 插入元素 删除元素 查找后继 ...

  7. 「算法笔记」树形 DP

    一.树形 DP 基础 又是一篇鸽了好久的文章--以下面这道题为例,介绍一下树形 DP 的一般过程. POJ 2342 Anniversary party 题目大意:有一家公司要举行一个聚会,一共有 \ ...

  8. 日均百万 PV 的站点如何做性能监测?试试「3M口罩」!

    对很多开发者而言,如果网站的日流量达到百万级别,峰值 PV 也突破了 3 万,这样的站点在线下测试的时候总是让人心力交瘁.... 生产环境下的性能监测问题更是尤其让人头疼! 开发同学在想,运维人员也在 ...

  9. LOJ #2542. 「PKUWC 2018」随机游走(最值反演 + 树上期望dp + FMT)

    写在这道题前面 : 网上的一些题解都不讲那个系数是怎么推得真的不良心 TAT (不是每个人都有那么厉害啊 , 我好菜啊) 而且 LOJ 过的代码千篇一律 ... 那个系数根本看不出来是什么啊 TAT ...

随机推荐

  1. 百度地图开发-引入地图SDK并配置 02

    百度地图开发-引入地图SDK并配置 02 通过上一篇文章的介绍,基本了解百度地图的基本信息,接下来就让我们一起来实际在项目中操作,显示出地图. 01 引入地图SDK 首先需要新建一个空白的Androi ...

  2. 第12篇-认识CodeletMark

    InterpreterCodelet依赖CodeletMark完成自动创建和初始化.CodeletMark继承自ResourceMark,允许自动析构,执行的主要操作就是,会按照Interpreter ...

  3. .net core2.1 迁移.net core 3.1

    1.解决方案->属性-->目标框架 .net core3.1 2.删除旧的Nuget包添加新的NuGet包 3.修改Startup.cs 修改ConfigureServices 修改Con ...

  4. 【springcloud】springcloud Greenwich SR4版本笔记

    springcloud Greenwich SR4版本笔记 本文只记录实际版本,配置,pom,代码以及注意事项.别的在其他springcloud 的F版本中已有详述. 示例代码地址:https://g ...

  5. 关于Ubuntu18.04 linux系统使用搜狗输入法 出现乱码

    解决: 执行下面的命令即可!无需重启系统 killall fcitx

  6. css - rem和vw

    css - rem和vw 物理像素 物理像素在不同的设备中1px里可以容纳的像素颗粒是不相同的,所以1px这个单位其实也是有N个像素颗粒填充的.同一尺寸屏幕的每个像素点所能容纳的像素颗粒越多显示效果越 ...

  7. css - 行高

    css - 行高 line-height行高 取值:px | em | rem | 百分比 | 纯数字 | normal | inherit 设置给:块.行内.行内块 应用给:文本 继承:块.行内.被 ...

  8. rasa form的中断形式 自然机器语言学习 人工智能

    Forms形式 最常见的对话模式之一是从用户那里收集一些信息以便做某事(预订餐厅.调用 API.搜索数据库等).这也称为**槽填充**. 用法# 要在 Rasa Open Source 中使用表单,您 ...

  9. Java中Scanner用法总结

    最近在做OJ类问题的时候,经常由于Scanner的使用造成一些细节问题导致程序不通过(最惨的就是网易笔试,由于sc死循环了也没发现,导致AC代码也不能通过...),因此对Scanner进行了一些总结整 ...

  10. 最长回文子序列---DP

    问题描述 给定一个字符串s,找到其中最长的回文子序列.可以假设s的最大长度为1000. 解题思路 1.说明 首先要弄清楚回文子串和回文子序列的区别,如果一个字符串是"bbbab", ...