poj 3358 Period of an Infinite Binary Expansion(数论:欧拉函数+快速幂取模)

本文主要是介绍poj 3358 Period of an Infinite Binary Expansion(数论:欧拉函数+快速幂取模),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

不太好理解题意的一道题

给出一个除式

要求找到对应二进制的循环起点和最小循环节长度

这里还考察了分数化小数的知识点。。。

这点不会难怪看题解都觉得很吃力尴尬

1/10

分数化小数的规律如下:

0.1 0.2 0.4 0.8 1.6 1.2 0.4(每次取左侧一位×2,如果大于10,小数位取1,再把这一位%10)

0    0    0   0    1    1    0

以1/10为例:1/10 2/10 4/10 8/10 16/10 32/10 64/10....

取模后:1/10 2/10 4/10 8/10 6/10 2/10 4/10 

这不就是个循环吗?循环节为4,循环起点为2,正好与题目相符。。如何去找循环节和循环起点?

由于是二进制,所以分子可以表示为2^x,而模数即q

2^x=2^y(mod q),2^x(2^(y-x)-1)=0(mod q),即p|2^x(2^(y-x)-1)

因为x^(y-1)-1为奇数,所以p|2^x

首先把q尽量整除2直到不能整除为止,这个步骤的次数就是满足原式最小的x,并得到q'。

2^(y-x)=1(mod q')

根据欧拉定理,t=y-x=phi(q')满足此式。

因为2^phi(q')和q'的最大公约数可能不为1

所以不一定是最小值,需要枚举phi(q')约数。

用int交就WA,用long long就过了

代码如下:

<span style="font-size:18px;">#include <cmath>
#include <vector>
#include <cstdio>
#include <iostream>
#include <algorithm>
#define LL long long
using namespace std;char str[100];
vector<LL> vec;LL gcd(LL a, LL b) {return b ? gcd(b, a%b) : a;
}LL euler_phi(LL n) {LL ans = n;LL m = sqrt(n+0.5);for(LL i=2; i<=m; ++i) {if(n%i == 0) {n /= i;ans = ans/i*(i-1);while(n%i == 0)n /= i;}}if(n > 1)ans = ans/n*(n-1);return ans;
}void get_fac(LL n) {LL m = sqrt(n+0.5);for(LL i=1; i<=m; ++i) {if(n%i == 0) {vec.push_back(i);if(n/i != i)vec.push_back(n/i);}}
}LL pow_mod(LL a, LL b, LL m) {LL ans = 1;while(b) {if(b & 1) {ans = ans*a%m;}a = a*a%m;b >>= 1;}return ans%m;
}int main(void) {char ch;LL cas = 0, tmp, x, y, p, q;while(scanf("%s", str) != EOF) {vec.clear();sscanf(str, "%lld%c%lld", &p, &ch, &q);if(!p) {printf("Case #%lld: 1,1\n", ++cas);continue;}//printf("p = %d\tq = %d\n", p, q);tmp = gcd(p, q);p /= tmp;q /= tmp;x = 1;while(q % 2 == 0) {q >>= 1;++x;}tmp = euler_phi(q);get_fac(tmp);sort(vec.begin(), vec.end());for(int i=0; i<vec.size(); ++i) {if(pow_mod(2, vec[i], q) == 1) {y = vec[i];break;}}printf("Case #%lld: %lld,%lld\n", ++cas, x, y);}return 0;
}</span>



这篇关于poj 3358 Period of an Infinite Binary Expansion(数论:欧拉函数+快速幂取模)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1059206

相关文章

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

Java function函数式接口的使用方法与实例

《Javafunction函数式接口的使用方法与实例》:本文主要介绍Javafunction函数式接口的使用方法与实例,函数式接口如一支未完成的诗篇,用Lambda表达式作韵脚,将代码的机械美感... 目录引言-当代码遇见诗性一、函数式接口的生物学解构1.1 函数式接口的基因密码1.2 六大核心接口的形态学

shell脚本快速检查192.168.1网段ip是否在用的方法

《shell脚本快速检查192.168.1网段ip是否在用的方法》该Shell脚本通过并发ping命令检查192.168.1网段中哪些IP地址正在使用,脚本定义了网络段、超时时间和并行扫描数量,并使用... 目录脚本:检查 192.168.1 网段 IP 是否在用脚本说明使用方法示例输出优化建议总结检查 1

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

Rust中的Option枚举快速入门教程

《Rust中的Option枚举快速入门教程》Rust中的Option枚举用于表示可能不存在的值,提供了多种方法来处理这些值,避免了空指针异常,文章介绍了Option的定义、常见方法、使用场景以及注意事... 目录引言Option介绍Option的常见方法Option使用场景场景一:函数返回可能不存在的值场景

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

电脑桌面文件删除了怎么找回来?别急,快速恢复攻略在此

在日常使用电脑的过程中,我们经常会遇到这样的情况:一不小心,桌面上的某个重要文件被删除了。这时,大多数人可能会感到惊慌失措,不知所措。 其实,不必过于担心,因为有很多方法可以帮助我们找回被删除的桌面文件。下面,就让我们一起来了解一下这些恢复桌面文件的方法吧。 一、使用撤销操作 如果我们刚刚删除了桌面上的文件,并且还没有进行其他操作,那么可以尝试使用撤销操作来恢复文件。在键盘上同时按下“C