• 多校联测13 菜


    题目大意

    有一个长度为 n n n的序列 a a a,你每次可以选择 gcd ⁡ \gcd gcd不为一的两个数 a i a_i ai a i + 1 a_{i+1} ai+1,将两个数合并,其值为两个数的 lcm \text{lcm} lcm,也就是删去 a i + 1 a_{i+1} ai+1,再让 a i = lcm ( a i , a i + 1 ) a_i=\text{lcm}(a_i,a_{i+1}) ai=lcm(ai,ai+1)

    求是否可以进行若干次操作,使得序列 a a a的长度变为 1 1 1。如果可行,就输出 Y e s Yes Yes;否则,输出 N o No No

    1 ≤ n ≤ 1 0 5 , 1 ≤ a i ≤ 700 1\leq n\leq 10^5,1\leq a_i\leq 700 1n105,1ai700


    题解

    首先,我们要知道,能合并的就合并,并不存在合并一个会使其他原本能合并的变得不能合并。我们可以维护一个栈,枚举每个元素,将当前元素与栈顶元素求 gcd ⁡ \gcd gcd,如果 gcd ⁡ \gcd gcd不为 1 1 1就取出栈顶元素并与当前元素合并,并继续判断接下来的栈顶元素与当前元素的 gcd ⁡ \gcd gcd是否为 1 1 1,直到不为一或栈为空。最后看栈内元素是否为 1 1 1即可。

    但在合并的过程中 a i a_i ai会很大,甚至连 i n t 128 int128 int128都可能存不下。而 700 700 700以内的质数只有 125 125 125个,我们可以用 b i t s e t bitset bitset来存 a i a_i ai有哪些质因子。

    虽然用了 b i t s e t bitset bitset,但还是要枚举每个 a i a_i ai是否有每个质因数,所以时间复杂度为 O ( n p ) O(np) O(np),其中 p p p 700 700 700以内质数的个数,也就是 125 125 125

    code

    #include
    using namespace std;
    const int N=700;
    int n,q1,a[100005],r[100005],z[705],p[705],q[100005];
    bitset<130>v[100005];
    void init(){
    	for(int i=2;i<=N;i++){
    		if(!z[i]) p[++p[0]]=i;
    		for(int j=1;j<=p[0]&&i*p[j]<=N;j++){
    			z[i*p[j]]=1;
    			if(i%p[j]==0) break;
    		}
    	}
    }
    int main()
    {
    	init();
    	scanf("%d",&n);
    	for(int i=1;i<=n;i++){
    		scanf("%d",&a[i]);
    		for(int j=1;a[i]>1&&j<=p[0];j++){
    			if(a[i]%p[j]==0) v[i][j-1]=1;
    			while(a[i]%p[j]==0) a[i]/=p[j];
    		}
    	}
    	for(int i=1;i<=n;i++){
    		q[++q1]=i;
    		while(q1>=2&&(v[q[q1]]&v[q[q1-1]]).count()){
    			v[q[q1-1]]|=v[q[q1]];
    			--q1;
    		}
    	}
    	if(q1==1) printf("Yes");
    	else printf("No");
    	return 0;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
  • 相关阅读:
    低代码平台中的“模型驱动”与“表单驱动”有何区别?
    python数据分析——数据分类汇总与统计
    对象的构造和析构
    STM32实战总结:HAL之电机
    Java 8 集合 Stream
    Java 地心地固坐标系转经纬度(WGS-84大地坐标)
    C Primer Plus(6) 中文版 第12章 存储类别、链接和内存管理 12.5 ANSI C类型限定符 12.6 关键概念 12.7 本章小结
    react-router 如何在组件外路由跳转
    Spring – 记录传入请求
    C++向指定内存地址写入数据(Windows)
  • 原文地址:https://blog.csdn.net/tanjunming2020/article/details/133815629