首先来看一个最简单的C语言实现质因数分解的列子:

#include

void main( )

{

int data, i = 2;

scanf("%d", &data);

while(data > 1)

{

if(data % i == 0)

{

printf("%d ", i);

data /= i;

}

else i++;

}

}

原理&&方法把一个合数分解为若干个质因数的乘积的形式,即求质因数的过程叫做分解质因数,分解质因数只针对合数

求一个数分解质因数,要从最小的质数除起,一直除到结果为质数为止。分解质因数的算式的叫短除法,和除法的性质差不多,还可以用来求多个个数的公因式:

以24为例:

2 -- 24

2 -- 12

2 -- 6

3 (3是质数,结束)

得出 24 = 2 × 2 × 2 × 3 = 2^3 * 3

代码

可先用素数筛选法,筛选出符合条件的质因数,然后for循环遍历即可,通过一道题目来show一下这部分代码

题目1

题目描述:

求正整数N(N>1)的质因数的个数。

相同的质因数需要重复计算。如120=2*2*2*3*5,共有5个质因数。

输入:

可能有多组测试数据,每组测试数据的输入是一个正整数N,(1

输出:

对于每组数据,输出N的质因数的个数。

样例输入:

120

样例输出:

5

提示:

注意:1不是N的质因数;若N为质数,N是N的质因数。

ac代码

#include

int main()

{

int n, count, i;

while (scanf("%d", &n) != EOF) {

count = 0;

for (i = 2; i * i <= n; i ++) {

if(n % i == 0) {

while (n % i == 0) {

count ++;

n /= i;

}

}

}

if (n > 1) {

count ++;

}

printf("%d\n", count);

}

return 0;

}

深入理解

我所谓的深入理解,就是通过4星的题目来灵活运用分解质因数的方法,题目如下

题目2

题目描述:

给定n,a求最大的k,使n!可以被a^k整除但不能被a^(k+1)整除。

输入:

两个整数n(2<=n<=1000),a(2<=a<=1000)

输出:

一个整数.

样例输入:

6 10

样例输出:

1

思路a^k和n!都可能非常大,甚至超过long long int的表示范围,所以也就不能直接用取余操作判断它们之间是否存在整除关系,因此我们需要换一种思路,从分解质因数入手,假设两个数a和b:

a = p1^e1 * p2^e2 * ... * pn^en, b = p1^d1 * p2^d2 * ... * pn^dn

, 则b除以a可以表示为:

b / a = (p1^d1 * p2^d2 * ... * pn^dn) / (p1^e1 * p2^e2 * ... * pn^en)

若b能被a整除,则 b / a必为整数,且两个素数必护质,则我们可以得出如下规律:

若a存在质因数px,则b必也存在该质因数,且该素因数在b中对应的幂指数必不小于在a中的幂指数

另b = n!, a^k = p1^ke1 * p2^ke2 * ... * pn^ken,因此我们需要确定最大的非负整数k即可。要求得该k,我们只需要依次测试a中每一个素因数,确定b中该素因数是a中该素因数的幂指数的多少倍即可,所有倍数中最小的那个即为我们要求得的k

分析到这里,剩下的工作似乎只是对a和n!分解质因数,但是将n!计算出来再分解质因数,这样n!数值太大。考虑n!中含有素因数p的个数,即确定素因数p对应的幂指数。我们知道n!包含了从1到n区间所有整数的乘积, 这些乘积中每一个p的倍数(包括其本身)都对n!贡献至少一个p因子,且我们知道在1到n中p的倍数共有n/p个。同理,计算p^2,p^3,...即可

代码

#include

#include

#include

#define N 1001

int prime[N], size;

/**

* 素数筛选法进行预处理

*/

void initProcess()

{

int i, j;

for (prime[0] = prime[1] = 0, i = 2; i < N; i ++) {

prime[i] = 1;

}

size = 0;

for (i = 2; i < N; i ++) {

if (prime[i]) {

size ++;

for (j = 2 * i; j < N; j += i) {

prime[j] = 0;

}

}

}

}

int main(void)

{

int i, n, a, k, num, count, base, tmp, *ansbase, *ansnum;

// 预处理

initProcess();

while (scanf("%d %d", &n, &a) != EOF) {

ansbase = (int *)calloc(size, sizeof(int));

ansnum = (int *)calloc(size, sizeof(int));

// 将a分解质因数

for (i = 2, num = 0; i < N && a != 1; i ++) {

if (prime[i] && a % i == 0) {

ansbase[num] = i;

ansnum[num] = 0;

while (a != 1 && a % i == 0) {

ansnum[num] += 1;

a = a / i;

}

num ++;

}

}

// 求最小的k

for (i = 0, k = 0x7fffffff; i < num; i ++) {

base = ansbase[i];

count = 0;

while (base <= n) {

count += n / base;

base *= ansbase[i];

}

tmp = count / ansnum[i];

if (tmp < k) k = tmp;

}

printf("%d\n", k);

}

return 0;

}

/**************************************************************

Problem: 1104

User: wangzhengyi

Language: C

Result: Accepted

Time:0 ms

Memory:916 kb

****************************************************************/

约数个数定理对于一个大于1的正整数n可以分解质因数:

n = p1^a1 * p2^a2 * p3^a3 * ... * pn^an

, 则n的正约数的个数为:

(a1 + 1) * (a2 + 1) * ... *(an + 1)

.其中p1,p2,..pn都是n的质因数,a1, a2...an是p1,p2,..pn的指数

证明

n可以分解质因数:n=p1^a1 * p2^a2 * p3^a3 * … * pk^ak,

由约数定义可知p1^a1的约数有:p1^0, p1^1, p1^2......p1^a1 ,共(a1+1)个;同理p2^a2的约数有(a2+1)个......pk^ak的约数有(ak+1)个

故根据乘法原理:n的约数的个数就是

(a1+1)*(a2+1)*(a3+1)*…* (ak+1)

题目3

题目描述:

输入n个整数,依次输出每个数的约数的个数

输入:

输入的第一行为N,即数组的个数(N<=1000)

接下来的1行包括N个整数,其中每个数的范围为(1<=Num<=1000000000)

当N=0时输入结束。

输出:

可能有多组输入数据,对于每组输入数据,

输出N行,其中每一行对应上面的一个数的约数的个数。

样例输入:

5

1 3 4 6 12

样例输出:

1

2

3

4

6

代码

#include

#include

#define N 40000

typedef long long int lint;

int prime[N], size;

void init()

{

int i, j;

for (prime[0] = prime[1] = 0, i = 2; i < N; i ++) {

prime[i] = 1;

}

size = 0;

for (i = 2; i < N; i ++) {

if (prime[i]) {

size ++;

for (j = 2 * i; j < N; j += i)

prime[j] = 0;

}

}

}

lint numPrime(int n)

{

int i, num, *ansnum, *ansprime;

lint count;

ansnum = (int *)malloc(sizeof(int) * (size + 1));

ansprime = (int *)malloc(sizeof(int) * (size + 1));

for (i = 2, num = 0; i < N && n != 1; i ++) {

if (prime[i] && n % i == 0) {

ansprime[num] = i;

ansnum[num] = 0;

while (n != 1 && n % i == 0) {

ansnum[num] += 1;

n /= i;

}

num ++;

}

}

if (n != 1) {

ansprime[num] = n;

ansnum[num] = 1;

num ++;

}

for (i = 0, count = 1; i < num; i ++) {

count *= (ansnum[i] + 1);

}

free(ansnum);

free(ansprime);

return count;

}

int main(void)

{

int i, n, *arr;

lint count;

init();

while (scanf("%d", &n) != EOF && n != 0) {

arr = (int *)malloc(sizeof(int) * n);

for (i = 0; i < n; i ++) {

scanf("%d", arr + i);

}

for (i = 0; i < n; i ++) {

count = numPrime(arr[i]);

printf("%lld\n", count);

}

free(arr);

}

return 0;

}

/**************************************************************

Problem: 1087

User: wangzhengyi

Language: C

Result: Accepted

Time:190 ms

Memory:1068 kb

****************************************************************/

您可能感兴趣的文章:

c语言作业 分解质因数,深入分析C语言分解质因数的实现方法相关推荐

  1. C语言作业二选择结构,C语言第二次作业参考答案选择结构.pdf

    1 C 语言第二次作业(选择结构)参考答案语言第二次作业(选择结构)参考答案 1.某年如果能被某年如果能被 4 整除整除,,但不能被但不能被 100 整除整除,,或者能被或者能被 400 整除的整除的 ...

  2. c语言作业系统输出超限,C语言网Online Judge系统支持语言和编译说明

    Online Judge系统支持语言和编译情况: 语言 编译器 语言版本 编译参数 C gcc 4.6.3 C99 gcc Main.c -o Main -Wall -lm –static -std= ...

  3. c语言作业朱鸣华,C语言程序设计教程

    本书介绍利用C语言进行计算机程序设计的基本知识.全书共分13章,详细介绍了C语言的基本概念.输入和输出.选择结构.循环结构.数组.函数.编译预处理.指针.结构体与共用体.文件等内容,同时介绍了面向对象 ...

  4. 优课在线C语言作业3,2018春C语言程序设计(苏小红)

    (1)自主式作业 由系统随机抽题的自主式作业,占总成绩的20%. 学生登录面向学生自主学习的高级语言能力测试平台(SSE),自主选择知识点和难度系数,平台自动从后台的题库系统中随机抽取相应知识点和难度 ...

  5. c语言作业自动批改代码,C语言助教用于批量改作业的脚本

    作为一名C语言助教,最恶心的事情莫过于改作业了,尤其是我们学校这种对输入输出都没有严格要求的题目,不能通过类似OJ的判题系统批量批改的,原则上是只能手动批改的.但是一直做着相同的简单的劳动真的很让人发 ...

  6. 华中科技大学c语言作业测评,华中科技大学C语言实验报告.docx

    华中科技大学C语言实验报告 课 程 实 验 报 告课程名称: C语言程序设计 专业班级: CS1409 学 号: U201414813 姓 名: 唐礼威 指导教师: 吴海 报告日期: 2015年6月2 ...

  7. 安徽大学计算机教学平台c语言作业,安徽大学计算机基础C语言选择题

    <安徽大学计算机基础C语言选择题>由会员分享,可在线阅读,更多相关<安徽大学计算机基础C语言选择题(5页珍藏版)>请在人人文库网上搜索. 1.1.能将高级语言编写的源程序转换成 ...

  8. c语言作业重庆科技学院,C语言程序设计学生上机报告-NO3.doc

    C语言程序设计学生上机报告-NO3.doc 重庆科技学院 上机实验报告(上机操作类) 课程名称 C 语 言 程 序 设 计 实验项目 循环结构程序设计(一) 机房名称 I313 上机时间 2017 年 ...

  9. mooc哈尔滨c语言作业答案,哈尔滨工业大学C语言2016年MOOC在线测试答案.doc

    哈尔滨工业大学C语言2016年MOOC在线测试答案 本答案仅供参考,请勿用于商业用途或进行未经同意的转载 第3周 1 计算两个数的平方和(3分) 题目内容: 从键盘读入两个实数,编程计算并输出它们的平 ...

  10. 哈工大c语言作业系统,2015秋C语言程序设计

    Q:本门课程面向的授课群体是什么?没有任何基础能学习本课程吗? A:本门课程的基本内容是面向零基础的学生.进阶内容是面向有一定编程基础或者大学计算机基础的学生,但是进阶内容不做为课程考试内容. Q:本 ...

最新文章

  1. 泼冷水:反思机器学习5年大跃进(附论文)
  2. java 右移两位_java左移右移运算符
  3. uni-app 实现小程序rsa加密(非对称加密原理)
  4. kerberos 下运行spark 报错 Requested user hdfs is banned
  5. 如何优雅地测量一只猫的体积?
  6. JQueryDOM之创建节点
  7. Linux 文件类型
  8. SVN刷新不及时,要手动操作
  9. 字符图形7-星号菱形
  10. php empty 0问题,解析:php empty 和空字符串区别
  11. LINK : fatal error LNK1561和LINK : fatal error LNK1168:解决方法
  12. android拒绝服务攻击,Android移动设备上的DDOS攻击
  13. 扣哒世界发布面向中小学阶段C++信奥课程体系
  14. [转]计算机视觉入门
  15. target找不到*.xml和*.properties文件 报错:FileNotFoundException
  16. 生而为人,请务必善良
  17. submit的用法实例
  18. 数字人民币银银合作以及平台接入的模式分析
  19. 铁电存储器FRAM的优劣势
  20. java json 二维数组_安卓中使用Gjson解析二维数组

热门文章

  1. oracle获取两个月前的时间
  2. 中文主播也能海外带货!同声传译助直播类应用开拓海外市场
  3. MP3参数,格式,术语有关一切内容详解。
  4. 技术债务-坏味道、Bug、漏洞
  5. mysql 快速复制数据,MySQL快速复制数据库的方法
  6. Sqlserver与Oracle 10g数据类型对照
  7. 显示网站Alexa世界排名的代码
  8. python 3d重建_python三维重建
  9. 微软中国招聘职位描述(英文)
  10. 浏览器调用html5播放器,html5视频播放器的浏览器兼容处理方法 - 小俊学习网