题意

给你一个长为 \(n\) 的环,你可以把它断成任意 \(k\) 段 \((1 < k \le n)\) ,使得每一段的 \(\gcd\) 都 \(>1\) 。

问总共有多少种方案,对于 \(10^9 + 7\) 取模。

\(n \le 10^5, 2 \le a_i \le 10^9\) 。

题解

首先我们考虑序列上怎么做。

我们令 \(dp_i\) 为到 \(i\) 这个点的方案数, \(pre_i\) 为 \(i\) 向前延伸最长的那个点满足 \((\displaystyle \gcd_{j = pre_i}^{i} a[j]) > 1\) 。

那么

\[\displaystyle dp_i = \sum_{j=pre_i-1}^{i-1} dp_j + [pre_i = 1]
\]

显然这个 \(dp\) 可以用前缀和来进行优化成 \(O(n)\) 的。

至于 \(pre_i\) 的处理可以用线段树求区间 \(\gcd\) ,然后用 two-pointers 来扫端点就行了,是 \(O(n( \log n + \log V))\) 的( \(\displaystyle V = \max_{i=1}^{n} a_i\) )。

好像利用 \(\gcd\) 的一些奇怪势能分析可以证明。


然后如果成环的话,我们只需要多考虑一种情况也就是 \(1,n\) 相连。

对于这种情况,可以考虑枚举最后面有 \(k\) 个数和 \(n\) 相连就行了。

然后每次计算的时候,可以类似于前面 \(dp\) 的计算就行了,但是要注意一下,那个 \([pre_i = 1]\) 要转化成后面 \(k\) 个数与前缀的 \(\gcd\) 是否 \(>1\) 。(也就是我们强行把后 \(k\) 个数当做一个整体提到前面就行了)

然后这样直接做是 \((n^2 \log V)\) 的,不够优秀。

但是我们发现很多数其实没有什么本质区别的,也就是后缀 \(\gcd\) 相同的一部分点可以一起计算。

这样的话,我们可以只在 \(\gcd\) 转折处,以及 \(0\) 号点计算就行了(因为要考虑上 \(1\) 向后那一片 \(\gcd\) 相同的数)。然后复杂度就是 \(O(n \log^2 V)\) 的了。

因为一个点向一端不断延伸,它的 \(\gcd\) 变换次数是 \(O(\log V)\) 的,因为每次变化至少会对于其中一个指数 \(-1\) 。

还有个特殊情况,也就是全部 \(\gcd > 1\) 的情况,需要将方案数 \(- (n - 1)\) 。(因为至少要分成 \(1\) 段)

最后就是 \(O(n (\log n + \log V) \log V)\) 的。

总结

对于一类划分环计数的题目,可以考虑枚举最后面那一段和前面相连的长度,然后直接一遍 \(O(n)\) 计数。

对于有些关于 \(\gcd\) 的题可以利用 \(\gcd\) 变换次数不超过 \(O(\log V)\) 来做。

代码

代码还是很好写的,可以参考一下。(其实博主参考了一下 ysgs 大佬的代码 )

#include <bits/stdc++.h>

#define For(i, l, r) for(register int i = (l), i##end = (int)(r); i <= i##end; ++i)
#define Fordown(i, r, l) for(register int i = (r), i##end = (int)(l); i >= i##end; --i)
#define Set(a, v) memset(a, v, sizeof(a))
#define Cpy(a, b) memcpy(a, b, sizeof(a))
#define debug(x) cout << #x << ": " << x << endl
#define DEBUG(...) fprintf(stderr, __VA_ARGS__) using namespace std; inline bool chkmin(int &a, int b) {return b < a ? a = b, 1 : 0;}
inline bool chkmax(int &a, int b) {return b > a ? a = b, 1 : 0;} inline int read() {
int x = 0, fh = 1; char ch = getchar();
for (; !isdigit(ch); ch = getchar()) if (ch == '-') fh = -1;
for (; isdigit(ch); ch = getchar()) x = (x << 1) + (x << 3) + (ch ^ 48);
return x * fh;
} void File() {
#ifdef zjp_shadow
freopen ("gcd-on-a-circle.in", "r", stdin);
freopen ("gcd-on-a-circle.out", "w", stdout);
#endif
} #define lson o << 1, l, mid
#define rson o << 1 | 1, mid + 1, r const int N = 1e5 + 1e3, Mod = 1e9 + 7; int a[N];
struct Segment_Tree { int Gcd[N << 2]; void Build(int o, int l, int r) {
if (l == r) { Gcd[o] = a[l]; return ; }
int mid = (l + r) >> 1;
Build(lson); Build(rson); Gcd[o] = __gcd(Gcd[o << 1], Gcd[o << 1 | 1]);
} int Query(int o, int l, int r, int ql, int qr) {
if (ql <= l && r <= qr) return Gcd[o];
int mid = (l + r) >> 1;
if (qr <= mid) return Query(lson, ql, qr);
if (ql > mid) return Query(rson, ql, qr);
return __gcd(Query(lson, ql, qr), Query(rson, ql, qr));
} } T; inline int Add(int a, int b) { return (a += b) >= Mod ? a - Mod : a; } int n, sum[N], dp[N], pre[N]; void Calc(int cur) {
sum[0] = 1;
For (i, 1, n) {
cur = __gcd(cur, a[i]);
if (cur > 1) dp[i] = sum[i - 1];
else dp[i] = Add(sum[i - 1], Mod - sum[max(0, pre[i] - 2)]);
sum[i] = Add(sum[i - 1], dp[i]);
}
} int main () { File(); n = read();
For (i, 1, n) a[i] = read(); T.Build(1, 1, n); pre[1] = 1;
For (i, 2, n)
for (pre[i] = pre[i - 1]; pre[i] < i; ++ pre[i])
if (T.Query(1, 1, n, pre[i], i) > 1) break; int Last = n, suf = 0, cur, ans = 0;
Fordown (i, n, 1) {
cur = __gcd(a[i], suf);
if (cur != suf || i == 1) {
Calc(suf); ans = Add(ans, Add(sum[Last], Mod - sum[i - 1]));
Last = i - 1; suf = cur;
}
} printf ("%d\n", Add(ans, (Mod - (T.Gcd[1] > 1 ? (n - 1) : 0)))); return 0;
}

CS Academy Gcd on a Circle(dp + 线段树)的更多相关文章

  1. ZOJ 3349 Special Subsequence 简单DP + 线段树

    同 HDU 2836 只不过改成了求最长子串. DP+线段树单点修改+区间查最值. #include <cstdio> #include <cstring> #include ...

  2. hdu 3016 dp+线段树

    Man Down Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  3. cf834D(dp+线段树区间最值,区间更新)

    题目链接: http://codeforces.com/contest/834/problem/D 题意: 每个数字代表一种颜色, 一个区间的美丽度为其中颜色的种数, 给出一个有 n 个元素的数组, ...

  4. Codeforces Round #620 F2. Animal Observation (hard version) (dp + 线段树)

    Codeforces Round #620 F2. Animal Observation (hard version) (dp + 线段树) 题目链接 题意 给定一个nm的矩阵,每行取2k的矩阵,求总 ...

  5. POJ1769 Minimizing maximizer(DP + 线段树)

    题目大概就是要,给一个由若干区间[Si,Ti]组成的序列,求最小长度的子序列,使这个子序列覆盖1到n这n个点. dp[i]表示从第0个到第i个区间且使用第i个区间,覆盖1到Ti所需的最少长度 对于Si ...

  6. [USACO2005][POJ3171]Cleaning Shifts(DP+线段树优化)

    题目:http://poj.org/problem?id=3171 题意:给你n个区间[a,b],每个区间都有一个费用c,要你用最小的费用覆盖区间[M,E] 分析:经典的区间覆盖问题,百度可以搜到这个 ...

  7. POJ 3162 Walking Race 树形DP+线段树

    给出一棵树,编号为1~n,给出数m 漂亮mm连续n天锻炼身体,每天会以节点i为起点,走到离i最远距离的节点 走了n天之后,mm想到知道自己这n天的锻炼效果 于是mm把这n天每一天走的距离记录在一起,成 ...

  8. 【uva1502/hdu4117-GRE Words】DP+线段树优化+AC自动机

    这题我的代码在hdu上AC,在uva上WA. 题意:按顺序输入n个串以及它的权值di,要求在其中选取一些串,前一个必须是后一个的子串.问d值的和最大是多少. (1≤n≤2×10^4 ,串的总长度< ...

  9. bzoj 1835 [ZJOI2010]base 基站选址(DP+线段树)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1835 [题意] 有n个村庄,每个村庄位于d[i],要求建立不多于k个基站,在第i个村庄 ...

随机推荐

  1. MySQL中varchar与char的区别以及varchar(50)中的50代表的涵义

     varchar与char的区别: 1).varchar与char的区别char是一种固定长度的类型,varchar则是一种可变长度的类型 尽可能的使用 varchar 代替 char ,因为首先变长 ...

  2. Elasticsearch之配置详解

    Cluster 集群名称,默认为elasticsearch: cluster.name: elasticsearch 设置一个节点的并发数量,有两种情况,一种是在初始复苏过程中: cluster.ro ...

  3. 怎么去掉Xcode工程中的某种类型的警告 Implicit conversion loses integer precision: 'NSInteger' (aka 'long') to 'int32

    unsigned long numComponents = CGColorGetNumberOfComponents([[UIColor blackColor] CGColor]); 2014年12月 ...

  4. 每周分享之JS数组的使用

    数组,一堆数字归为一组,就是一个数组,一堆对象放在一个组里,也是一个数组,概念很容易懂,说白了就是一个有限集合. JS数组的语法无法两种,插入和移除(语法自行科普).用处挺常见的,既然数组是一个集合, ...

  5. 【问题解决方案】之 hadoop 用jps命令后缺少namenode的问题

    用Xshell连接腾讯cloud里的虚拟机后,jps命令查无namenode导致过滤排序程序跑不起来,如图: 解决方案: Google之,说需要重启,格式化后再启动Hadoop.但鉴于本人不知道实现的 ...

  6. 自签名证书 nginx tomcat

    给Nginx配置一个自签名的SSL证书 - 廖雪峰的官方网站 https://www.liaoxuefeng.com/article/0014189023237367e8d42829de24b6eaf ...

  7. asp.net mvc Dateset读取Excel数据

    //处理Excel //读取Excel [NonAction] public static DataSet ExcelToDS(string Path) { //根据情况选择合适的连接字符,参考msd ...

  8. js压箱底的宝贝

    框架的确好用, 不过他们也隐藏了JavaScript中丑陋的细节和DOM的运作机制. 如果你的目标是敢于自称"我懂JavaScript", 那么花时间学习框架无异于南辕北辙. 下面 ...

  9. Laravel5 创建自定义门面(Facade)

    门面为应用服务容器中的绑定类提供了一个“静态”接口.Laravel 内置了很多门面,你可能在不知道的情况下正在使用它们.Laravel 的门面作为服务容器中底层类的“静态代理”,相比于传统静态方法,在 ...

  10. [转帖]前端-chromeF12 谷歌开发者工具详解 Network篇

    前端-chromeF12 谷歌开发者工具详解 Network篇 https://blog.csdn.net/qq_39892932/article/details/82493922 blog 也是原作 ...