题目背景

题目描述

给定一长度为n的动态序列,请编写一种数据结构,要求支持m次操作,包括查询序列中一闭区间中所有数的GCD,与对一闭区间中所有数加上或减去一个值。

输入输出格式

输入格式:

第1行两个数n,m,表示序列长度和操作次数。

第2行n个数ai,表示给定序列。

第3行至第m+2行,每行3~4个数:

(1) 1 x y k 表示将[x,y]上的所有数加上k。

(2) 2 x y 表示询问[x,y]上所有数的GCD。

输出格式:

对所有操作2,输出一个数,表示询问结果。

输入输出样例

输入样例#1:

7 3
4 8 2 6 5 7 10
2 1 4
1 2 3 7
2 2 3
输出样例#1:

2
3

说明

定义:a,b∈Z时,gcd(a,b)=gcd(abs(a),abs(b))

对于30%的数据,n,m<=1000。

对于90%的数据,n,m<=100000。

对于100%的数据,n,m<=200000,ai<=1e7(初始),abs(k)<=1e7。

题解:

如果题目要求改为只支持区间查询,那么线段树或ST表都可以很方便地实现。进一步思考,区间修改无法用普通线段树实现的根本原因在于对[l,r]修改后[l,r]的结果无法O(1)计算出来。
如果区间修改改为单点修改,则可以用线段树暴力log(n)修改。

此处证明一个引理:gcd(a1,a2,a3,...,ai)=gcd(a1,a2-a1,a3-a2,...ai-ai-1).
设S为ai的公因数集合,T为ai-ai-1的公因数集合
设p为ai的任意一个公因数,则有p|ai,由整除的性质知p|ai-ai-1,则p一定是ai-ai-1的公因数,所以S是T的子集。
同理,设q为ai-ai-1的任意一个公因数,运用同样的性质可知q一定是ai的公因数,所以T是S的子集。
综上,S=T,所以max{S}=max{T},即gcd(a1,a2,a3,...,ai)=gcd(a1,a2-a1,a3-a2,...ai-ai-1).

所以我们将原数组a进行差分,设差分后数组为d,区间查询[l,r]则转化为gcd(gcd(d[l+1,r]),a[l]);差分后区间修改变为单点修改,可用线段树暴力实现。

具体操作:将原数组进行差分,用一棵支持单点修改的线段树维护gcd,将差分数组用一个树状数组维护前缀和(用来求出变化后的a[l],也可以合并在线段树中)。
注意:差分时对区间[l,r]涉及到对r+1的操作,为防止溢出,线段树区间增大至[1,n+1]。

代码如下:

#include<bits/stdc++.h>
#define LL long long
#define lowbit(x) x&(-x)
using namespace std;
const int maxn=2e5+10;
LL node[4*maxn],a[maxn],c[maxn],d[maxn];
int n,m;LL ans;
LL gcd(LL a,LL b){return b==0?a:gcd(b,a%b);}
void pushup(int x){node[x]=abs(gcd(node[x<<1],node[x<<1|1]));}
void build(int x,int l,int r)
{
    if(l==r){node[x]=d[l];return;}
    int mid=(l+r)>>1;
    build(x<<1,l,mid);build(x<<1|1,mid+1,r);
    pushup(x);
}
void change(int x,int l,int r,int pos,int d)
{
    if(l==r){node[x]+=d;return;}
    int mid=(l+r)>>1;
    if(pos<=mid){change(x<<1,l,mid,pos,d);}
    else{change(x<<1|1,mid+1,r,pos,d);}
    pushup(x);
}
void query(int x,int l,int r,int sj,int tj)
{
    if(sj<=l&&r<=tj){ans=abs(gcd(node[x],ans));return;}
    int mid=(l+r)>>1;
    if(sj<=mid){query(x<<1,l,mid,sj,tj);}
    if(mid+1<=tj){query(x<<1|1,mid+1,r,sj,tj);}
    pushup(x);
}
void add(int x,int d)
{
    int i;
    for(i=x;i<=n;i+=lowbit(i)){c[i]+=d;}
}
LL sum(int x)
{
    int i;LL ans=0;
    for(i=x;i>=1;i-=lowbit(i)){ans+=c[i];}
    return ans;
}
int main()
{
    int i,j,flag,l,r,dlt;
    cin>>n>>m;
    for(i=1;i<=n;i++){scanf("%lld",&a[i]);}
    n++;
    for(i=1;i<=n;i++){d[i]=a[i]-a[i-1];add(i,d[i]);}
    build(1,1,n);
    //for(i=1;i<=3*n;i++){printf("i=%d node[i]=%d\n",i,node[i]);}
    for(i=1;i<=m;i++)
    {
        scanf("%d%d%d",&flag,&l,&r);
        if(flag==1){scanf("%d",&dlt);change(1,1,n,l,dlt);change(1,1,n,r+1,-dlt);add(l,dlt);add(r+1,-dlt);}
        else{ans=0;query(1,1,n,l+1,r);/*printf("ans=%d sum(l)=%d\n",ans,sum(l));*/printf("%lld\n",abs(gcd(ans,sum(l))));}
    }
    return 0;
}

Luogu T9376 区间GCD的更多相关文章

  1. HDU 5726 GCD 区间GCD=k的个数

    GCD Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submis ...

  2. Codeforces 914D - Bash and a Tough Math Puzzle 线段树,区间GCD

    题意: 两个操作, 单点修改 询问一段区间是否能在至多一次修改后,使得区间$GCD$等于$X$ 题解: 正确思路; 线段树维护区间$GCD$,查询$GCD$的时候记录一共访问了多少个$GCD$不被X整 ...

  3. bzoj 5028: 小Z的加油店——带修改的区间gcd

    Description 小Z经营一家加油店.小Z加油的方式非常奇怪.他有一排瓶子,每个瓶子有一个容量vi.每次别人来加油,他会让 别人选连续一段的瓶子.他可以用这些瓶子装汽油,但他只有三种操作: 1. ...

  4. dutacm.club 1094: 等差区间(RMQ区间最大、最小值,区间GCD)

    1094: 等差区间 Time Limit:5000/3000 MS (Java/Others)   Memory Limit:163840/131072 KB (Java/Others)Total ...

  5. HDU5381【莫队算法+区间GCD特性】

    前言: 主要最近在刷莫队的题,这题GCD的特性让我对莫队的使用也有了新的想法.给福利:神犇的一套莫队算法题 先撇开题目,光说裸的一个莫队算法,主要的复杂度就是n*sqrt(n)对吧,这里我忽略了一个左 ...

  6. 区间 GCD

    区间 GCD题目描述最近 JC 同学刚学会 gcd,于是迷上了与 gcd 有关的问题.今天他又出了一道这样的题目,想要考考你,你能顺利完成吗?给定一个长度为 n 的字符串 s[1..n],串仅包含小写 ...

  7. HDU 5726 GCD (2016多校、二分、ST表处理区间GCD、数学)

    题目链接 题意 : 给出一个有 N 个数字的整数数列.给出 Q 个问询.每次问询给出一个区间.用 ( L.R ) 表示.要你统计这个整数数列所有的子区间中有多少个和 GCD( L ~ R ) 相等.输 ...

  8. 区间gcd

    http://codeforces.com/problemset/problem/914/D 题意:给你n个数,两种操作:1.询问区间[l,r]在至多一次修改一个数的条件下区间gcd是否等于x. 2. ...

  9. 区间加值,区间gcd, 牛客949H

    牛客小白月赛16H 小阳的贝壳 题目链接 题意 维护一个数组,支持以下操作: 1: 区间加值 2: 询问区间相邻数差的绝对值的最大值 3: 询问区间gcd 题解 设原数组为\(a\), 用线段树维护\ ...

随机推荐

  1. flask为多个接口添加同一个拦截器的方法

    前言 最近又抽掉出来写一个 Python 项目, 框架使用 Flask , 又有些新心得, 比如本篇所说, 想要将某个蓝图加上统一的权限控制, 比如 admin 蓝图全部有一个统一的拦截器判断是否有权 ...

  2. 借助Docker搭建JMeter+Grafana+Influxdb监控平台

    我们都知道Jmeter提供了原生的结果查看,既然有原生的查看结果,为什么还要多此一举使用其他工具进行查看呢,除了查看内容丰富外还有最主要的原因:Jmeter提供的查看结果插件本身是比较消耗性能的,所以 ...

  3. sort方法和sorted()函数

    sort方法和sorted()函数的区别: 相同点:都能完成排序操作. 不同点: (1)使用sort()方法对list排序会修改list本身,不会返回新list,sort()不能对dict字典进行排序 ...

  4. for update语句锁机制问题

    数据库小知识学习系列 问题: MySQL InnoDB中,select where xxx=123 for update:该xxx没有索引,是使用表锁还是全部数据加行锁? 答: InnoDB引擎(默认 ...

  5. 解压rpm文件

    rpm2cpio zabbix-2.2.2-0.el6.zbx.src.rpm |cpio -div

  6. LeetCode383. 赎金信

    题目 给定一个赎金信 (ransom) 字符串和一个杂志(magazine)字符串,判断第一个字符串 ransom 能不能由第二个字符串 magazines 里面的字符构成.如果可以构成,返回 tru ...

  7. 【一天一个知识点系列】- Http之状态码

    状态码 简介 HTTP 状态码负责表示客户端 HTTP 请求的返回结果. 标记服务器端的处理是否正常. 通知出现的错误等工作 作用及类别 作用:状态码告知从服务器端返回的请求结果 状态码的类别 注意: ...

  8. Redis 实战 —— 02. Redis 简单实践 - 文章投票

    需求 功能: P15 发布文章 获取文章 文章分组 投支持票 数值及限制条件 P15 如果一篇文章获得了至少 200 张支持票,那么这篇文章就是一篇有趣的文章 如果这个网站每天有 50 篇有趣的文章, ...

  9. 图解 ECDHE 密钥交换算法

    HTTPS 常用的密钥交换算法有两种,分别是 RSA 和 ECDHE 算法. 其中,RSA 是比较传统的密钥交换算法,它不具备前向安全的性质,因此现在很少服务器使用的.而 ECDHE 算法具有前向安全 ...

  10. Sklearn 与 TensorFlow 机器学习实战—一个完整的机器学习项目

    本章中,你会假装作为被一家地产公司刚刚雇佣的数据科学家,完整地学习一个案例项目.下面是主要步骤: 项目概述. 获取数据. 发现并可视化数据,发现规律. 为机器学习算法准备数据. 选择模型,进行训练. ...