题目链接

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

分析 :

首先对于给出一个区间要你给出 GCD

这个操作可以使用线段树来做、线段树是可以维护 GCD 的

但是由于这题的静态区间 (即数列里面的数不会被改变)

那么也有另外一种方法来回答区间 GCD 的问询

预处理的复杂度是 O(nlogn) 、问询是 O(1)

即 ST表、类似处理 RMQ 问题那样子把代码改成求 GCD 就行了

其次它还要你给出和问询区间 GCD 一样的子区间的个数

考虑预处理出所有出现的 GCD 到底在多少个不同的子区间出现过

用 map 存一下也能够做到复杂度在 O(nlogn) 内

这个的处理需要用到二分技巧、还有一点数学知识

首先如果固定区间左端点、那么右端点越大、则 GCD 必定单调不增

所以可以考虑枚举所有的位置作为左端点

然后通过二分的方式找出所有以左端点为开头的 GCD 一样的区间

例如 2 4 6 5 1

枚举 2 作为左端点时候、那么第一次二分会二分到 6 的位置

即 2 4 6 的 GCD 都是 2、则 mp[2] += pos(6) - pos(2) + 1 = 3 - 1 + 1 = 3

然后将 GCD 改变一下变成 GCD = gcd( GCD, 5 )

此时 GCD 会变成 1 、那么第二次二分就会二分到 1 的位置

所以 mp[1] = pos(1) - pos(5) + 1 = 2

接下来就以 4 为左端点、以此类推........

但是你可能会有忧虑、即使是这样子的二分、会不会因为二分次数太多超时

那么你考虑一下、对于端点 L 、其数值是 arr[L]

那么以它为左端点的区间的 GCD 必定是 arr[L] 质因子的某些乘积组合

每加入一个能够改变 GCD 的数、则 GCD 必定减少至少两倍

质因子的数量的 log 的、那么每次二分必定不超过 log(1e9) 次

#include<bits/stdc++.h>
#define LL long long
#define ULL unsigned long long

#define scl(i) scanf("%lld", &i)
#define scll(i, j) scanf("%lld %lld", &i, &j)
#define sclll(i, j, k) scanf("%lld %lld %lld", &i, &j, &k)
#define scllll(i, j, k, l) scanf("%lld %lld %lld %lld", &i, &j, &k, &l)

#define scs(i) scanf("%s", i)
#define sci(i) scanf("%d", &i)
#define scd(i) scanf("%lf", &i)
#define scIl(i) scanf("%I64d", &i)
#define scii(i, j) scanf("%d %d", &i, &j)
#define scdd(i, j) scanf("%lf %lf", &i, &j)
#define scIll(i, j) scanf("%I64d %I64d", &i, &j)
#define sciii(i, j, k) scanf("%d %d %d", &i, &j, &k)
#define scddd(i, j, k) scanf("%lf %lf %lf", &i, &j, &k)
#define scIlll(i, j, k) scanf("%I64d %I64d %I64d", &i, &j, &k)
#define sciiii(i, j, k, l) scanf("%d %d %d %d", &i, &j, &k, &l)
#define scdddd(i, j, k, l) scanf("%lf %lf %lf %lf", &i, &j, &k, &l)
#define scIllll(i, j, k, l) scanf("%I64d %I64d %I64d %I64d", &i, &j, &k, &l)

#define lson l, m, rt<<1
#define rson m+1, r, rt<<1|1
#define lowbit(i) (i & (-i))
#define mem(i, j) memset(i, j, sizeof(i))

#define fir first
#define sec second
#define VI vector<int>
#define ins(i) insert(i)
#define pb(i) push_back(i)
#define pii pair<int, int>
#define VL vector<long long>
#define mk(i, j) make_pair(i, j)
#define all(i) i.begin(), i.end()
#define pll pair<long long, long long>

#define _TIME 0
#define _INPUT 0
#define _OUTPUT 0
clock_t START, END;
void __stTIME();
void __enTIME();
void __IOPUT();
using namespace std;

;

int N, Q;
int arr[maxn];
];
int idx[maxn];
map<int, LL> mp;

inline void init_idx()
{
    idx[] = -;
    ; len<=N; len++)
        idx[len] = ((len & (len-)) == ) ? idx[len-] +  : idx[len-];
}

inline void init_rmq()
{
    ; j<=; j++){
        ; i+(<<(j-))<= N; i++){
            dp[i][j] = __gcd(dp[i][j-], dp[i+(<<(j-))][j-]);
        }
    }
}

int RMQ(int L, int R)
{
    ];
    <<k)+][k]);
}

inline void init_gcd_num()
{
    ; i<=N; i++){
        int j = i;
        int GCD = arr[j];
        while(j <= N){
            int L, R, pos;
            L = pos = j;
            R = N;
            while(L <= R){
                );
                , pos = mid;
                ;
            }
            mp[GCD] += 1LL * (pos - j + );
            j = pos + ;
            GCD = RMQ(i, j);
        }
    }
}

int main(void){__stTIME();__IOPUT();

    ;
    sci(nCase);

    while(nCase--){

        mp.clear();

        sci(N);

        ; i<=N; i++)
            sci(arr[i]),
            dp[i][] = arr[i];

        init_idx();
        init_rmq();
        init_gcd_num();

        sci(Q);

        printf("Case #%d:\n", ++Case);

        while(Q--){
            int L, R;
            scii(L, R);
            int GCD = RMQ(L, R);
            printf("%d %lld\n", GCD, mp[GCD]);
        }
    }

__enTIME();;}

void __stTIME()
{
    #if _TIME
        START = clock();
    #endif
}

void __enTIME()
{
    #if _TIME
        END = clock();
        cerr<<"execute time = "<<(double)(END-START)/CLOCKS_PER_SEC<<endl;
    #endif
}

void __IOPUT()
{
    #if _INPUT
        freopen("in.txt", "r", stdin);
    #endif
    #if _OUTPUT
        freopen("out.txt", "w", stdout);
    #endif
}

HDU 5726 GCD (2016多校、二分、ST表处理区间GCD、数学)的更多相关文章

  1. HDU 5727 Necklace ( 2016多校、二分图匹配 )

    题目链接 题意 : 给出 2*N 颗珠子.有 N 颗是阴的.有 N 颗是阳的.现在要把阴阳珠子串成一个环状的项链.而且要求珠子的放置方式必须的阴阳相间的.然后给出你 M 个限制关系.格式为 ( A.B ...

  2. HDU5726 GCD(二分 + ST表)

    题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5726 Description Give you a sequence of N(N≤100, ...

  3. Codeforces 475D CGCDSSQ 区间gcd值

    题目链接 题意 给定一个长度为 \(n\) 的数列 \(a_1,...,a_n\) 与 \(q\) 个询问 \(x_1,...,x_q\),对于每个 \(x_i\) 回答有多少对 \((l,r)\) ...

  4. 2016多校联合训练1 D题GCD (ST表+二分)

    暑假颓废了好久啊...重新开始写博客 题目大意:给定10w个数,10w个询问.每次询问一个区间[l,r],求出gcd(a[l],a[l+1],...,a[r])以及有多少个区间[l',r']满足gcd ...

  5. HDU 5726 GCD (RMQ + 二分)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5726 给你n个数,q个询问,每个询问问你有多少对l r的gcd(a[l] , ... , a[r]) ...

  6. HDU 5726 GCD(RMQ+二分)

    http://acm.split.hdu.edu.cn/showproblem.php?pid=5726 题意:给出一串数字,现在有多次询问,每次询问输出(l,r)范围内所有数的gcd值,并且输出有多 ...

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

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

  8. HDU 5726 GCD

    传送门 GCD Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Problem ...

  9. HDU 5726 GCD(DP)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=5726 [题目大意] 给出数列An,对于询问的区间[L,R],求出区间内数的GCD值,并且求出GCD ...

随机推荐

  1. h5中的分组元素figure、figcaption、hgroup元素介绍

    分组元素用于对页面中的内容进行分组. figure元素和figcaption元素 figure元素用于定义独立的流内容(图像.图表.照片.代码等),一般指一个独立的单元.figure元素的内容应该与主 ...

  2. MySQ-表关系-外键-修改表结构-复制表-03

    目录 前言 不合理的表结构(案例) 带来的问题 如何解决问题? 如何确定表关系? 表关系 一对多 多对多 一对一 应用场景 判断表关系最简单的语法 三种关系常见案例 如何建立表关系? 外键 forei ...

  3. MySQL中导入Excel表格中的数据

    在数据库中建立好响应的数据库.表(参考excel表格中列中的名字和内容): 将excel表格另存为txt文件,选择“文本文件(制表符分割)”: 打开相应的txt文件,只留下要导入的数据(windows ...

  4. 《深入理解 Java 虚拟机》学习 -- 垃圾收集器

    <深入理解 Java 虚拟机>学习 -- 垃圾收集器 1. Serial 收集器(新生代) 含义: 单线程收集器. 缺点: 进行垃圾收集时,必须暂停其他所有的工作线程. 优点: 简单而高效 ...

  5. ModbusTCP报文详解【一】

    [1]功能码01H [2]功能码02H [3]功能码03H [4]功能码04H

  6. kali安装dnsdict6

    https://src.fedoraproject.org/lookaside/pkgs/thc-ipv6/thc-ipv6-2.7.tar.gz/2975dd54be35b68c140eb2a6b8 ...

  7. href="javascript:show_login()"意思

    整句话意味着当你点击一个超链接时,你会触发函数show_login. Href是一个超链接,通过单击该超链接触发. javascript:后面是JS代码 show_login():表示JS的函数的油烟 ...

  8. 微信小程序点击复制功能

    wx.setClipboardData({ data: '这是要复制的文字', success: function (res) { wx.showModal({ title: '提示', conten ...

  9. 【Day1】2.安装运行Python

     视频地址(全部) https://edu.csdn.net/course/detail/26057 课件地址(全部) https://download.csdn.net/download/gentl ...

  10. DA_05_Linux(CentOS6.7) 安装MySql5.7数据库

    1系统约定 安装文件下载目录:/data/software Mysql目录安装位置:/usr/local/mysql 数据库保存位置:/data/mysql 日志保存位置:/data/log/mysq ...