【算法学习笔记】31:试除法分解质因数及求解欧拉函数

试除法分解质因数

原理

O ( n ) O(\sqrt{n}) O(n )时间内可以对 n n n分解质因数,原理就是从最小质因数2开始遍历,第一个遍历到的因数就是2,一定是一个质因数。

每次都从 n n n里将遇到的质因子除干净,那么下次遇到的因数 i i i也一定是一个质因数。

可以用反证法:如果下次遇到的因数 i i i不是一个质因数,那么说明有 < i <i <i的另一个质因数 j j j,那么因为我们每次遇到质因数都会从 n n n里除干净,所以 j < i j < i j<i一定在过去的某一轮被除干净了,矛盾,证毕。

这个遍历只需要在 i ⋅ i ≤ n i \cdot i \leq n iin的情况下继续就可以了,因为一旦 i ⋅ i > n i \cdot i > n ii>n,最后剩下的要么 n n n是1,要么 n n n是最后一个质因数(且出现了一次方),一定不存在其它没遍历到的质因数了。

可以用反证法:假如 n n n还是一个合数,还存在没遍历到的质因子 i i i,那么 n i \frac{n}{i} in也一定是一个因数,这个因数的任何一个质因子都 < i < i <i,又一定从 n n n里被剔除过了,矛盾,证毕。

最后记得判断一下 n n n是不是剩下一个一次方的最大质因子,也就是 n n n如果不是1就是最后的那个质因数。

例题:AcWing 867. 分解质因数

#include <iostream>using namespace std;int main() {int t;cin >> t;while (t -- ) {int n;cin >> n;for (int i = 2; i * i <= n; i ++ ) {int mi = 0;while (n % i == 0) {mi ++ ;n /= i;}if (mi) cout << i << " " << mi << endl;}if (n != 1) cout << n << " " << 1 << endl;cout << endl;}return 0;
}

试除法求解欧拉函数

利用试除法分解质因数,同样可以解决欧拉函数的计算问题。

定义与计算公式

欧拉函数 ϕ ( n ) \phi(n) ϕ(n)表示从1到n中和n互质的数字的个数,两个数字除了1没有其它公因数就是互质。比如从1到6之间和6互质的数字只有1和5,所以 p h y ( 6 ) = 2 phy(6) = 2 phy(6)=2

n n n分解质因数之后的结果是:
n = p 1 α 1 + p 2 α 2 + . . + p k α k n = p_1^{\alpha_1} + p_2^{\alpha_2} + .. + p_k^{\alpha_k} n=p1α1+p2α2+..+pkαk
那么欧拉公式的计算公式是:
ϕ ( n ) = n ⋅ ( 1 − 1 p 1 ) ⋅ ( 1 − 1 p 2 ) ⋅ . . . ⋅ ( 1 − 1 p k ) \phi(n) = n \cdot (1 - \frac{1}{p_1}) \cdot (1 - \frac{1}{p_2}) \cdot ... \cdot (1 - \frac{1}{p_k}) ϕ(n)=n(1p11)(1p21)...(1pk1)

证明

思路是用容斥原理。假设 n n n的质因子只有 p 1 p_1 p1 p k p_k pk,那么根据欧拉函数的定义,只要从1到n中去掉 p 1 p_1 p1 p k p_k pk所有数的倍数,最后剩下的就是和 n n n互质的数。

第一步:减去所有 p 1 p_1 p1 p k p_k pk的倍数

n − ⌊ n p 1 ⌋ − ⌊ n p 2 ⌋ − . . . − ⌊ n p k ⌋ n - \lfloor \frac{n}{p_1} \rfloor - \lfloor \frac{n}{p_2} \rfloor - ... - \lfloor \frac{n}{p_k} \rfloor np1np2n...pkn

但是因为某个 p i p_i pi的倍数同时可能也是 p j p_j pj的倍数,所以直接去掉每个数的倍数就会存在多去除的情况,因此就要把多去掉的部分加回来。

第二步:加上所有 p i ⋅ p j p_i \cdot p_j pipj的倍数

+ ⌊ n p 1 ⋅ p 2 ⌋ + ⌊ n p 1 ⋅ p 3 ⌋ + . . . + ⌊ n p k − 1 ⋅ p k ⌋ + \lfloor \frac{n}{p_1 \cdot p_2} \rfloor + \lfloor \frac{n}{p_1 \cdot p_3} \rfloor + ... + \lfloor \frac{n}{p_{k-1} \cdot p_k} \rfloor +p1p2n+p1p3n+...+pk1pkn

进而考虑,如果有一个数是 p i p_i pi p j p_j pj p k p_k pk的倍数,那么在第一步会被 p i p_i pi p j p_j pj p k p_k pk各减去一次,第二步会被 p i ⋅ p j p_i \cdot p_j pipj p j ⋅ p k p_j \cdot p_k pjpk p i ⋅ p k p_i \cdot p_k pipk各加上一次,减掉三次加上三次,没有加没有减,所以下一步要减去所有同时是三个数的倍数。

第三步:减去所有 p i ⋅ p j ⋅ p k p_i \cdot p_j \cdot p_k pipjpk的倍数

− ⌊ n p 1 ⋅ p 2 ⋅ p 3 ⌋ − ⌊ n p 1 ⋅ p 2 ⋅ p 4 ⌋ − . . . − ⌊ n p k − 2 ⋅ p k − 1 ⋅ p k ⌋ - \lfloor \frac{n}{p_1 \cdot p_2 \cdot p_3} \rfloor - \lfloor \frac{n}{p_1 \cdot p_2 \cdot p_4} \rfloor - ... - \lfloor \frac{n}{p_{k-2} \cdot p_{k-1} \cdot p_k} \rfloor p1p2p3np1p2p4n...pk2pk1pkn

所以接下来就是加上所有四个质数乘积分之 n n n,再减去所有五个质数乘积分之 n n n

将前面给出的欧拉公式的计算公式展开,就可以发现和这个加加减减的式子是相等的,即:
ϕ ( n ) = n − ⌊ n p 1 ⌋ − ⌊ n p 2 ⌋ − . . . − ⌊ n p k ⌋ + ⌊ n p 1 ⋅ p 2 ⌋ + ⌊ n p 1 ⋅ p 3 ⌋ + . . . + ⌊ n p k − 1 ⋅ p k ⌋ − ⌊ n p 1 ⋅ p 2 ⋅ p 3 ⌋ − ⌊ n p 1 ⋅ p 2 ⋅ p 4 ⌋ − . . . − ⌊ n p k − 2 ⋅ p k − 1 ⋅ p k ⌋ . . . = n ⋅ ( 1 − 1 p 1 ) ⋅ ( 1 − 1 p 2 ) ⋅ . . . ⋅ ( 1 − 1 p k ) \begin{align} \phi(n) &= n - \lfloor \frac{n}{p_1} \rfloor - \lfloor \frac{n}{p_2} \rfloor - ... - \lfloor \frac{n}{p_k} \rfloor \\ & + \lfloor \frac{n}{p_1 \cdot p_2} \rfloor + \lfloor \frac{n}{p_1 \cdot p_3} \rfloor + ... + \lfloor \frac{n}{p_{k-1} \cdot p_k} \rfloor \\ & - \lfloor \frac{n}{p_1 \cdot p_2 \cdot p_3} \rfloor - \lfloor \frac{n}{p_1 \cdot p_2 \cdot p_4} \rfloor - ... - \lfloor \frac{n}{p_{k-2} \cdot p_{k-1} \cdot p_k} \rfloor \\ & ... \\ & = n \cdot (1 - \frac{1}{p_1}) \cdot (1 - \frac{1}{p_2}) \cdot ... \cdot (1 - \frac{1}{p_k}) \end{align} ϕ(n)=np1np2n...pkn+p1p2n+p1p3n+...+pk1pknp1p2p3np1p2p4n...pk2pk1pkn...=n(1p11)(1p21)...(1pk1)

用眼睛看也能看出上下两个式子相等,首先把上面式子里的 n n n提取出来,就是下面式子最外面的 n n n。然后上面式子里的开头1就是下面式子乘法展开时候每一项选择前面的1的结果。上面式子里的 1 p i \frac{1}{p_i} pi1就是下面式子里只有 ( 1 − 1 p i ) (1 - \frac{1}{p_i}) (1pi1)选择了 − 1 p i - \frac{1}{p_i} pi1,其它式子都选择1的结果,因此符号一定是负的。上面式子里的 1 p i ⋅ p j \frac{1}{p_i \cdot p_j} pipj1就是下面式子里只有 ( 1 − 1 p i ) (1 - \frac{1}{p_i}) (1pi1) ( 1 − 1 p j ) (1 - \frac{1}{p_j}) (1pj1)选择了 − 1 p i - \frac{1}{p_i} pi1 − 1 p j - \frac{1}{p_j} pj1,其它式子都选择1的结果,因此符号一定是正的。以此类推。

例题:AcWing 873. 欧拉函数

根据前面推导出的计算公式,只要用最开始的 n n n,每次抠出质因数 p i p_i pi,然后乘上 ( 1 − 1 p i ) (1 - \frac{1}{p_i}) (1pi1),这里可以改写成除以 p i p_i pi再乘以 p i − 1 p_i - 1 pi1,避免了浮点运算的同时也防止乘爆了溢出。

#include <iostream>using namespace std;int phi(int n) {int res = n;for (int i = 2; i * i <= n; i ++ ) {if (n % i) continue;res = res / i * (i - 1);while (n % i == 0) n /= i;}if (n != 1) res = res / n * (n - 1);return res;
}int main() {int t; cin >> t;while (t -- ) {int n; cin >> n;cout << phi(n) << endl;}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/2196.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

9.7 visual studio 搭建yolov10的onnx的预测(c++)

1.环境配置 在进行onnx预测前&#xff0c;需要搭建的环境如下: 1.opencv环境的配置&#xff0c;可参考博客:9.2 c搭建opencv环境-CSDN博客 2.libtorch环境的配置&#xff0c;可参考博客&#xff1a;9.4 visualStudio 2022 配置 cuda 和 torch (c)-CSDN博客 3.cuda环境的配置…

自建RustDesk服务器

RustDesk服务端 下面的截图是我本地的一个服务器做为演示用&#xff0c;你自行的搭建服务需要该服务器有固定的ip地址 1、通过宝塔面板快速安装 2、点击【安装】后会有一个配置信息&#xff0c;默认即可 3、点击【确认】后会自动安装等待安装完成 4、安装完成后点击【打开…

浅谈云计算15 | 存储可靠性技术(RAID)

存储可靠性技术 一、存储可靠性需求1.1 数据完整性1.2 数据可用性1.3 故障容错性 二、传统RAID技术剖析2.1 RAID 02.2 RAID 12.3 RAID 52.4 RAID 62.5 RAID 10 三、RAID 2.0技术3.1 RAID 2.0技术原理3.1.1 两层虚拟化管理模式3.1.2 数据分布与重构 3.2 RAID 2.0技术优势3.2.1 自…

Android JecPack组件之LifeCycles 使用详解

一、背景 LifeCycle 是一个可以感知宿主生命周期变化的组件。常见的宿主包括 Activity/Fragment、Service 和 Application。LifeCycle 会持有宿主的生命周期状态的信息&#xff0c;当宿主生命周期发生变化时&#xff0c;会通知监听宿主的观察者。 LifeCycle 的出现主要是为了…

Facebook 隐私风波:互联网时代数据安全警钟

在社交媒体飞速发展的今天&#xff0c;个人数据的隐私保护已成为全球关注的焦点。作为全球最大的社交平台之一&#xff0c;Facebook面临的隐私问题&#xff0c;尤其是数据泄露事件&#xff0c;频繁引发公众的广泛讨论。从用户信息被滥用到数据泄漏&#xff0c;Facebook的隐私挑…

candb++ windows11运行报错,找不到mfc140.dll

解决问题记录 mfc140.dll下载 注意&#xff1a;放置位置别搞错了

蓝桥杯备赛:顺序表和单链表相关算法题详解(上)

目录 一.询问学号&#xff08;顺序表&#xff09; 1.题目来源&#xff1a; 2.解析与代码实现&#xff1a; &#xff08;1&#xff09;解析&#xff1a; &#xff08;2&#xff09;代码实现&#xff1a; 二.寄包柜&#xff08;顺序表&#xff09; 1.题目来源&#xff1a; …

uni-app的学习

uni-app 有着跨平台支持、丰富的插件和生态系统、高性能、集成开发工具HBuilderX的配合使用。允许使用者仅通过一套代码发布到多平台使用。 uni-app官网 uni-app 是一个适合开发跨平台移动应用和小程序的框架&#xff0c;能够大幅提高开发效率。 一、了解 1.1 工具准备 从Git…

基于光偏振与光学调制实现白光干涉相移

基于光的偏振特性和一些光学元件对光的调制作用&#xff0c;实现白光干涉中的光学相移原理是一个复杂而精细的过程。以下是对这一原理的详细解释&#xff1a; 一、光的偏振特性 光的偏振是指光波在传播过程中&#xff0c;光矢量的方向和大小有规则变化的现象。圆偏振光的电场…

Flutter:封装ActionSheet 操作菜单

演示效果图 action_sheet_util.dart import package:ducafe_ui_core/ducafe_ui_core.dart; import package:flutter/material.dart; import package:demo/common/index.dart;class ActionSheetUtil {/// 底部操作表/// [context] 上下文/// [title] 标题/// [items] 选项列表 …

使用yarn命令创建Vue3项目

文章目录 1.技术栈2.创建流程2.1创建vue3项目2.2选择配置项2.3进入项目目录 3.使用Yarn启动项目3.1安装依赖3.2运行项目 1.技术栈 yarnvitevue3 2.创建流程 2.1创建vue3项目 vue create 项目名称2.2选择配置项 直接回车可选择Vue3 2.3进入项目目录 cd 项目名称默认在当前…

【Node.js的安装与配置】

目录&#xff1a; 一&#xff1a;下载Node.js二&#xff1a;安装Node.js三&#xff1a;配置存放目录四&#xff1a;配置环境变量五&#xff1a;配置淘宝镜像六&#xff1a;测试Node.js 一&#xff1a;下载Node.js &#x1f534; 下载地址&#xff1a;https://www.nodejs.com.cn…

【AIGC】SYNCAMMASTER:多视角多像机的视频生成

标题&#xff1a;SYNCAMMASTER: SYNCHRONIZING MULTI-CAMERA VIDEO GENERATION FROM DIVERSE VIEWPOINTS 主页&#xff1a;https://jianhongbai.github.io/SynCamMaster/ 代码&#xff1a;https://github.com/KwaiVGI/SynCamMaster 文章目录 摘要一、引言二、使用步骤2.1 TextT…

左神算法基础提升--1

文章目录 哈希函数哈希函数的主要特点确定性快速计算输出长度固定离散性 哈希表哈希表的原理解题 布隆过滤器布隆过滤器的主要特点高效性快速查询空间效率误报率 布隆过滤器的原理 一致性哈希一致性哈希原理一致性哈希应用 哈希函数 哈希函数是一种将任意长度的输入&#xff0…

【Go】Go Gin框架初识(一)

1. 什么是Gin框架 Gin框架&#xff1a;是一个由 Golang 语言开发的 web 框架&#xff0c;能够极大提高开发 web 应用的效率&#xff01; 1.1 什么是web框架 web框架体系图&#xff08;前后端不分离&#xff09;如下图所示&#xff1a; 从上图中我们可以发现一个Web框架最重要…

VS Code 的扩展下载安装的最新方式

离线包的下载 在 2024年的时候&#xff0c;在VS Code的官方扩展市场&#xff1a;https://marketplace.visualstudio.com/ &#xff0c; 搜索到需要的扩展之后&#xff0c;是可以在对应的页面现在最新版本和几个历史版本的扩展的安装包。 下载下来的扩展包的文件是后缀是 vsix …

【Vue3 入门到实战】3. ref 和 reactive区别和适用场景

目录 ​编辑 1. ref 部分 1.1 ref定义基本数据类型 1.2 ref 定义引用数据类型 2. reactive 函数 3. ref 和 reactive 对比 3.1 原理 3.2 区别 3.3 使用原则 在 Vue 3 中 ref 和 reactive 是用于创建响应式数据的两个核心函数。它们都属于 Composition API 的一部分&…

浅谈云计算07 | 云安全机制

云计算安全机制 一、引言二、加密技术&#xff1a;数据的隐形护盾三、散列机制&#xff1a;数据完整性的忠诚卫士四、数字签名&#xff1a;数据来源与真伪的鉴定专家五、公钥基础设施&#xff08;PKI&#xff09;&#xff1a;信任的基石六、身份与访问管理&#xff08;IAM&…

【Sql递归查询】Mysql、Oracle、SQL Server、PostgreSQL 实现递归查询的区别与案例(详解)

文章目录 Mysql 5.7 递归查询Mysql 8 实现递归查询Oracle递归示例SQL Server 递归查询示例PostgreSQL 递归查询示例 更多相关内容可查看 Mysql 5.7 递归查询 MySQL 5.7 本身不直接支持标准 SQL 中的递归查询语法&#xff08;如 WITH RECURSIVE 这种常见的递归查询方式&#xf…

【Unity3D】【已解决】TextMeshPro无法显示中文的解决方法

TextMeshPro无法显示中文的解决方法 现象解决方法Assets 目录中新建一个字体文件夹在C:\Windows\Fonts 中随便找一个中文字体的字体文件把字体文件拖到第一步创建的文件夹中右键导入的字体&#xff0c;Create---TextMeshPro---Font Asset&#xff0c;创建字体文件资源把 SDF文件…