【ACM数论】和式变换技术,也许是最好的讲解之一

在做数论题时,往往需要进行和式变换,然后变换成我们可以处理的和式,再针对和式做筛法、整除分块等操作。

本文将介绍一些常见的和式变换技术。

以下出现的概念大部分为个人总结,未必是学术界/竞赛界的统一说法,有不严谨的地方请谅解。

? 作者:Eriktse
? 简介:19岁,211计算机在读,现役ACM银牌选手?力争以通俗易懂的方式讲解算法!❤️欢迎关注我,一起交流C++/Python算法。(优质好文持续更新中……)?
? 原文链接(阅读原文获得更好阅读体验):https://www.eriktse.com/algorithm/1101.html

和式的基本形式

和式一般有两种:区间枚举型整除枚举型

区间枚举型

我们的前缀和就是一个典型的区间枚举型和式。

假设我们有一个定义域为\(x\in[1, n],x\in Z^+\)的函数\(f(x)\),那么我们可以设一个前缀和函数\(F(x)\),定义为:

\[F(x) = \sum_{i=1}^{x}f(i) = f(1) + f(2) + ... + fx()
\]

求和符号中,如果没有特殊说明,一般枚举的都是整数,且步长为1。

整除枚举型

约数个数是一个典型的整除枚举型和式,我们可以容易的写出它的表达式:

\[f(n) = \sum_{d|n}1
\]

其中 \(d|n\) 表示 \(i\) 可以整除 \(n\) ,即 \(i\)\(n\) 的因子。

约数之和也是一个整除枚举型和式,表达式如下:

\[g(n) = \sum_{d|n}d
\]

和式的基本性质

可拆分性质

第一种拆分如下:

\[\sum_{i=1}^{n}a_i = \sum_{i=1}^{m}a_i + \sum_{i=m+1}^{n}a_i
\]

这是显然的,但是基本上用不着。

第二种拆分如下:

\[\sum_{i=1}^{n}(a_i + b_i) = \sum_{i=1}^{n}a_i + \sum_{i=1}^{n}b_i
\]

这也是显然的。

常数可提取

当我们的和式里面乘上了一个常数\(k\),那么这个常数是可以提出来的,由于我们讨论的数域是整数域,这个\(k\)一般为整数。(其实对于实数也是满足条件的)。

\[\sum_{i=1}^{n}ka_i = k\sum_{i=1}^{n}a_i
\]

整除枚举型变换为区间枚举型(重要)

就比如上面那个约数之和的函数:

\[g(i) = \sum_{d|n}d = \sum_{i=1}^{n}[d|n]
\]

我们知道\(d\)的取值一定在\([1, n]\),所以我们可以转换枚举类型,此时枚举指标的范围就要改变,同时加上一个布尔函数来限定。

我们称枚举的东西为“指标”,例如上面和式中d|n中的di=1中的i

指标变换(重要)

给定一个整数\(k\),对于下面这种和式,我们可以把指标进行转换。

\[\sum_{i=1}^{n}\sum_{j=1}^{n}[gcd(i, j) = k]
\]

现在令\(i = i'k,j=j'k\),为什么会这么想呢?因为我们后面的布尔函数中要求\(i, j\)都包含因子\(k\),如果枚举的\(i, j\)不是\(k\)的倍数的时候这个式子是没有贡献的。

所以我们可以不一个个枚举\(i, j\),变为枚举\(k\)的倍数。我们进行整体的代换:

\[\sum_{i'k = 1}^{n}\sum_{j'k=1}^{n}[gcd(i'k, j'k) = k]
\]

然后变换枚举范围和布尔函数,注意这里\(i\)的起点本应该是\(\lfloor\frac{1}{k}\rfloor\),但是\(0\)是没有讨论意义的所以我们从\(1\)开始。

\[\sum_{i=1}^{\lfloor\frac{n}{k}\rfloor}\sum_{j=1}^{\lfloor\frac{n}{k}\rfloor}[gcd(i, j) = 1]
\]

现在我们可以发现后面这个布尔函数就变成了一个常见的积性函数\(\epsilon\),接下来就可以通过公式\(\mu * I = \epsilon\)进行莫比乌斯反演(其中符号\(*\)表示狄利克雷卷积)。

交换求和次序(重要)

上式进行莫比乌斯反演后可以得到如下的和式(如果不懂莫比乌斯反演可以暂时先不管,之后再学),设\(m=\lfloor\frac{n}{k}\rfloor\)

\[\sum_{i=1}^{m}\sum_{j=1}^{m}\sum_{d|gcd(i, j)}\mu(d)
\]

我们可以发现\(d|gcd(i, j)\)这个条件等价于\([d|i][d|j]\),即\(d\)同时是\(i\)\(j\)的因子。

接下来我们进行一次枚举类型的转换:

\[\sum_{i=1}^{m}\sum_{j=1}^{m}\sum_{d=1}^{m}[d|i][d|j]\mu(d)
\]

接下来我们将\(d\)的求和符号从后面换到前面去,因为在\(\mu(d)\)中没有包含\(i, j\)的内容,可以直接换,这里需要自己理解一下。

\[\\sum_{d=1}^{m}\mu(d)\sum_{i=1}^{m}[d|i]\sum_{j=1}^{m}[d|j]
\]

转换为整除分块形式(十分重要)

上式转换完成后,我们可以发现后面两坨是可以进行整除分块的。

\[\sum_{i=1}^{m}[d|i] = \lfloor\frac{m}{d}\rfloor
\]

怎么理解呢?这个式子表达的就是当\(d\)确定了,在区间[1, n]中有多少整数是\(d\)的倍数,显然是\(\lfloor\frac{m}{d}\rfloor\)个。

那么和式就可转换为:

\[\sum_{i=1}^{m}\lfloor\frac{m}{d}\rfloor\lfloor\frac{m}{d}\rfloor
\]

例题

luogu P2257 YY的GCD:https://www.luogu.com.cn/problem/P2257

阅读题意我们可以知道题目所求为,不妨设\(n\le m\)

\[ans=\sum_{i=1}^{n}\sum_{j=1}^{m}[gcd(i,j)\in prim]
\]

接下来开始变换:

\[\sum_{i=1}^{n}\sum_{j=1}^{m}\sum_{p\in prim}[gcd(i,j)=p]
\]

\[\sum_{p\in prim}\sum_{i=1}^{\lfloor\frac{n}{p}\rfloor}\sum_{j=1}^{\lfloor\frac{m}{p}\rfloor}[gcd(i,j)=1]
\]

莫比乌斯反演:

\[\sum_{p\in prim}\sum_{i=1}^{\lfloor\frac{n}{p}\rfloor}\sum_{j=1}^{\lfloor\frac{m}{p}\rfloor}\sum_{d|gcd(i,j)}\mu(d)
\]

注意这里\(n\le m\),接着变换。

\[\sum_{p\in prim}\sum_{i=1}^{\lfloor\frac{n}{p}\rfloor}\sum_{j=1}^{\lfloor\frac{m}{p}\rfloor}\sum_{d=1}^{\lfloor\frac{n}{p}\rfloor}[d|i][d|j]\mu(d)
\]

\[\sum_{p\in prim}\sum_{d=1}^{\lfloor\frac{n}{p}\rfloor}\mu(d)\sum_{i=1}^{\lfloor\frac{n}{p}\rfloor}[d|i]\sum_{j=1}^{\lfloor\frac{m}{p}\rfloor}[d|j]
\]

后面两坨可以进行整除分块,同时换一下\(p\)的枚举类型:

\[\sum_{p=1}^{n}[p\in prim]\sum_{d=1}^{\lfloor\frac{n}{p}\rfloor}\mu(d)\lfloor\frac{n}{pd}\rfloor\lfloor\frac{m}{pd}\rfloor
\]

\(T=pd\),交换求和次序。

\[\sum_{p=1}^{n}[p\in prim][p|T]\sum_{T=1}^{n}\mu(\frac{T}{p})\lfloor\frac{n}{T}\rfloor\lfloor\frac{m}{T}\rfloor
\]

再交换求和次序:

\[\sum_{T=1}^{n}\lfloor\frac{n}{T}\rfloor\lfloor\frac{m}{T}\rfloor\sum_{p=1}^{n}[p\in prim][p|T]\mu(\frac{T}{p})
\]

现在发现\(p\)后面那一块,可以通过类似欧拉筛的方法进行预处理。

我们设一个函数:

\[F(T) = \sum_{p=1}^{n}[p \in prim][p|T]\mu(\frac{T}{p})
\]

那么\(F(T)\)的含义就是对于\(T\)的每一个质因子\(p\),将它的\(\mu(\frac{T}{p})\)加到自身上。

做完了。

Code:

#include <bits/stdc++.h>
#define int long long
using namespace std;
const int N = 1e7 + 9;

int sum[N], mu[N];

void init(int n = N - 2)
{
	bitset<N> vis;
	vector<int> prim;
	vis[1] = true;
	mu[1] = 1;
	
	for(int i = 2;i <= n; ++ i)
	{
		if(!vis[i])prim.push_back(i), mu[i] = -1;
		
		for(int j = 0;j < prim.size() and i * prim[j] <= n; ++ j)
		{
			vis[i * prim[j]] = true;
			if(i % prim[j] == 0)break;//此时i * prim[j]含有平方因子
			
			mu[i * prim[j]] = -mu[i];//此时i * prim[j]的本质不同质因子+1,或已经含有平方因子
		}
	}
	
	for(int i = 0;i < prim.size(); ++ i)
	{
		for(int j = 1; prim[i] * j  <= n; ++ j)
		{
			sum[prim[i] * j] += mu[j];
		}
	}
	
	for(int i = 1;i <= n; ++ i)sum[i] += sum[i - 1];
	
}

void solve()
{
	int n, m;scanf("%lld %lld", &n, &m);
	if(n > m)swap(n, m);
	int ans = 0;
	for(int l = 1, r;l <= n; l = r + 1)
	{
		r = min(n / (n / l), m / (m / l));
		ans += (sum[r] - sum[l - 1]) * (n / l) * (m / l);
	}
	printf("%lld\n", ans);
}

signed main()
{
	init();
	int _;scanf("%lld", &_);
	while(_ --)solve();
	return 0;
}

结束

? 本文由eriktse原创,创作不易,如果对您有帮助,欢迎小伙伴们点赞?、收藏⭐、留言?

原文链接:https://www.cnblogs.com/eriktse/p/17280451.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:【ACM数论】和式变换技术,也许是最好的讲解之一 - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • 回溯理论基础及leetcode例题

    学习参考 回溯 与递归相辅相成;回溯是递归的副产品,只要有递归就会有回溯。回溯函数也就是递归函数,指的都是一个函数。 回溯搜索法 纯暴力搜索解决的问题 组合问题:N个数里面按一定规则找出k个数的集合切割问题:一个字符串按一定规则有几种切割方式子集问题:一个N个数的集合里有多少符合条件的子集排列问题:N个数按一定规则全排列,有几种排列方式(与组合差别,排列有元…

    算法与数据结构 2023年4月17日
    00
  • C语言详细分析结构体的内存对齐规则

    C语言详细分析结构体的内存对齐规则 1. 什么是内存对齐 在计算机内存中,每个数据都需要分配一定的内存空间存储,这些空间的大小不一定相同。内存对齐就是要求每个数据按照某个规则,分配其所需的内存空间。 在C语言中,结构体是一种复合数据类型,由多个数据成员组成。结构体的数据成员排列顺序、数据类型均可能不同,因此需要内存对齐来规定内存空间的分配。 2. C语言中结…

    数据结构 2023年5月17日
    00
  • NDK 数据结构之队列与栈等的实现

    NDK 数据结构之队列与栈等的实现 引言 Android NDK 是 Android 开发工具包的一部分,可以用 C 和 C++ 编写应用程序和库。NDK 带来了许多好处,例如可以针对不同的平台进行优化,可以通过调用底层 C/C++ 库实现更高效的算法等。 在本篇文档中,我们将探讨如何使用 NDK 实现一些基础的数据结构,包括队列、栈等等。 队列的实现 队列…

    数据结构 2023年5月17日
    00
  • C语言程序设计第五版谭浩强课后答案(第二章答案)

    首先,需要说明的是本题涉及到一个特定的知识领域,即C语言程序设计,以及该领域内某个具体教材的课后习题解答。因此,本攻略的重心将放在如何利用Markdown格式对该领域内的知识进行准确、清晰的表达和展示上。 下面是本攻略的目录: C语言程序设计第五版谭浩强课后答案(第二章答案)攻略 一、简介 二、题目列表 三、示例说明 示例一 示例二 四、总结 一、简介 本攻…

    数据结构 2023年5月17日
    00
  • K最近邻算法(KNN)—sklearn+python实现方式

    以下是关于“K最近邻算法(KNN)—sklearn+python实现方式”的完整攻略: 简介 K最近邻算法(KNN)是一种用于分类和回归的机器学习算法,它可以根据最近的K个邻居来预测新数据点的标签或值。在本教程中,我们将介绍如何使用Python和sklearn库实现KNN算法,并提供两个示例说明。 实现KNN算法 以下是使用Python和sklearn库…

    python 2023年5月14日
    00
  • 基于python的Paxos算法实现

    基于Python的Paxos算法实现 Paxos算法是一种分布式一致性算法,它可以保证在分布式系统中的多个节点之间达成一致的决策。本文将介绍如何使用Python实现Paxos算法,并提供两个示例说明。 算法原理 Paxos算法的核心思想是通过多个节点之间的协商和投票来达成一致的决策。在Pax算法中,有三种角色:提议者、接受者和学习者。提议者提出一个提议,接受…

    python 2023年5月14日
    00
  • C语言数据结构二叉树先序、中序、后序及层次四种遍历

    C语言数据结构二叉树四种遍历 什么是二叉树 二叉树是一种非常重要的数据结构,在计算机科学中具有广泛的应用。它由节点和边组成,每个节点最多有两个子节点。二叉树有许多种遍历方法,可以用来查找节点、在树中插入新节点、删除节点等操作。 二叉树遍历 二叉树遍历是指对二叉树的节点进行访问,有4种遍历方式: 先序遍历(Preorder Traversal) 中序遍历(In…

    数据结构 2023年5月17日
    00
  • C语言植物大战数据结构希尔排序算法

    C语言植物大战数据结构希尔排序算法 什么是希尔排序 希尔排序是一种基于插入排序的排序算法,也叫做“缩小增量排序”。和插入排序不同的是,希尔排序的插入排序是对一定间隔的元素进行插入排序,而不是对数组中相邻的元素进行排序。 希尔排序的流程和方法 希尔排序的主要流程是根据元素间的间隔d,分组进行插入排序,依次减小d值。最后当d=1的时候,再按照插入排序的方法对整个…

    数据结构 2023年5月17日
    00
合作推广
合作推广
分享本页
返回顶部