[BestCoder Round #3] hdu 4910 Problem about GCD

2024-01-28 13:18

本文主要是介绍[BestCoder Round #3] hdu 4910 Problem about GCD,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

转载自:

http://www.kuangbin.net/archives/hdu4910  慢慢学习.

HDU 4910

Problem about GCD

求小于等于 N 的与N互质的所有数的乘积MOD N

 

Problem about GCD

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 26    Accepted Submission(s): 6


Problem Description
Given integer m. Find multiplication of all 1<=a<=m such gcd(a, m)=1 (coprime with m) modulo m.

 

Input
Input contains multiple tests, one test per line.
Last line contains -1, it should be skipped. [Technical Specification]
m <= 10^18

 

Output
For each test please output result. One case per line. Less than 160 test cases.

 

Sample Input
1 2 3 4 5 -1

 

Sample Output
0 1 2 3 4

 

Source
BestCoder Round #3

 

很显然,需要找规律。 答案要么是N-1,要么是1.

答案为N-1的情况是: 1 2 4

此外 如果 N!=4 && N%4 == 0  答案是1

如果N为偶数,要求N/2 是 p^k的形式;

如果N为奇数,要求N是p^k的形式。

/* ***********************************************
Author        :kuangbin
Created Time  :2014/8/3 19:24:03
File Name     :D.cpp
************************************************ */#include <stdio.h>
#include <string.h>
#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
#include <set>
#include <map>
#include <string>
#include <math.h>
#include <stdlib.h>
#include <time.h>
using namespace std;/* ************************************************** Miller_Rabin 算法进行素数测试* 速度快,可以判断一个 < 2^63 的数是不是素数***************************************************/const int S = 20; //随机算法判定次数,一般8~10就够了// 计算ret = (a*b)%c     a,b,c < 2^63
long long mult_mod(long long a,long long b,long long c)
{a %= c;b %= c;long long ret = 0;long long tmp = a;while(b){if(b & 1){ret += tmp;if(ret > c)ret -= c;//直接取模慢很多}tmp <<= 1;if(tmp > c)tmp -= c;b >>= 1;}return ret;
}// 计算 ret = (a^n)%mod
long long pow_mod(long long a,long long n,long long mod)
{long long ret = 1;long long temp = a%mod;while(n){if(n & 1)ret = mult_mod(ret,temp,mod);temp = mult_mod(temp,temp,mod);n >>= 1;}return ret;
}// 通过 a^(n-1)=1(mod n)来判断n是不是素数
// n-1 = x*2^t 中间使用二次判断
// 是合数返回true, 不一定是合数返回false
bool check(long long a,long long n,long long x,long long t)
{long long ret = pow_mod(a,x,n);long long last = ret;for(int i = 1;i <= t;i++){ret = mult_mod(ret,ret,n);if(ret == 1 && last != 1 && last != n-1)return true;//合数last = ret;}if(ret != 1)return true;else return false;
}
//**************************************************
// Miller_Rabin算法
// 是素数返回true,(可能是伪素数)
// 不是素数返回false
//**************************************************
bool Miller_Rabin(long long n)
{if( n < 2)return false;if( n == 2)return true;if( (n&1) == 0)return false;//偶数long long x = n - 1;long long t = 0;while( (x&1)==0 ){x >>= 1; t++;}srand(time(NULL));/* *************** */for(int i = 0;i < S;i++){long long a  = rand()%(n-1) + 1;if( check(a,n,x,t) )return false;}return true;
}//**********************************************
// pollard_rho 算法进行质因素分解
//
//
//*********************************************
long long factor[100];//质因素分解结果(刚返回时时无序的)
int tol;//质因素的个数,编号0~tol-1long long gcd(long long a,long long b)
{long long t;while(b){t = a;a = b;b = t%b;}if(a >= 0)return a;else return -a;
}//找出一个因子
long long pollard_rho(long long x,long long c)
{long long i = 1, k = 2;srand(time(NULL));long long x0 = rand()%(x-1) + 1;long long y = x0;while(1){i ++;x0 = (mult_mod(x0,x0,x) + c)%x;long long d = gcd(y - x0,x);if( d != 1 && d != x)return d;if(y == x0)return x;if(i == k){y = x0; k += k;}}
}//对 n进行素因子分解,存入factor. k设置为107左右即可
void findfac(long long n,int k)
{if(n == 1)return;if(Miller_Rabin(n)){factor[tol++] = n;return;}long long p = n;int c = k;while( p >= n)p = pollard_rho(p,c--);//值变化,防止死循环kfindfac(p,k);findfac(n/p,k);
}int main()
{//freopen("in.txt","r",stdin);//freopen("out.txt","w",stdout);long long n;while(scanf("%I64d",&n) == 1){if(n == -1)break;if(n <= 1){printf("0\n");continue;}if(n == 1 || n == 2 || n == 4){printf("%I64d\n",n-1);continue;}if(n != 4 && n % 4 == 0){printf("1\n");continue;}tol = 0;if(n%2 == 0){findfac(n/2,107);sort(factor,factor+tol);bool flag = true;for(int i = 1;i < tol;i++)if(factor[i-1] != factor[i]){flag = false;break;}if(flag)printf("%I64d\n",n-1);else printf("1\n");}else{findfac(n,107);sort(factor,factor+tol);bool flag = true;for(int i = 1;i < tol;i++)if(factor[i-1] != factor[i]){flag = false;break;}if(flag)printf("%I64d\n",n-1);else printf("1\n");}}return 0;
}


这篇关于[BestCoder Round #3] hdu 4910 Problem about GCD的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

hdu 2093 考试排名(sscanf)

模拟题。 直接从教程里拉解析。 因为表格里的数据格式不统一。有时候有"()",有时候又没有。而它也不会给我们提示。 这种情况下,就只能它它们统一看作字符串来处理了。现在就请出我们的主角sscanf()! sscanf 语法: #include int sscanf( const char *buffer, const char *format, ... ); 函数sscanf()和

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

uva 10025 The ? 1 ? 2 ? ... ? n = k problem(数学)

题意是    ?  1  ?  2  ?  ...  ?  n = k 式子中给k,? 处可以填 + 也可以填 - ,问最小满足条件的n。 e.g k = 12  - 1 + 2 + 3 + 4 + 5 + 6 - 7 = 12 with n = 7。 先给证明,令 S(n) = 1 + 2 + 3 + 4 + 5 + .... + n 暴搜n,搜出当 S(n) >=

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d