约数的个数 链接:
点击打开链接
时间限制:1秒
空间限制:65536K
热度指数:14731
算法知识视频讲解
题目描述
输入n个整数,依次输出每个数的约数的个数
输入描述:
输入的第一行为N,即数组的个数(N<=1000) 接下来的1行包括N个整数,其中每个数的范围为(1<=Num<=1000000000) 当N=0时输入结束。
输出描述:
可能有多组输入数据,对于每组输入数据, 输出N行,其中每一行对应上面的一个数的约数的个数。
示例1
输入
5 1 3 4 6 12
输出
1 2 3 4 6
思路分析:
方法1:利用约数个数定理,用一个数组记录各个质因数的指数ri,再计算(r1+1)*(r2+1)*......
记得遍历时遍历到sqrt(n),来降低时间复杂度,不然会超时。
方法2:设置一个全局变量div记录因子的个数。对于大于1的整数至少一个因子,所以div初值为1;
如果一个数能被他的某一个因数整除,那么div+=2;因为s=a*b,s%a==0,a,b都为因子 ;
遍历时遍历到sqrt(n),来降低时间复杂度,不然会超时;
需要再判断输入的这个数是否大于1,如果比1大,那么还有一个因数就是他本身,div+=1;如果该数就等于1,那么div保持不变;
AC代码(方法1):
#include <iostream>
#include<cstring>
#include<cmath>
using namespace std;
const int nmax=1e5;
int num[nmax];//记录各个输入的数
int nd[nmax];//存储每个数的质因数的指数
void cal(int n){
memset(nd,0,sizeof(nd));
int len=0;
int sq=sqrt(n);
for(int i=2;i<=sq;i++){//如若直接遍历到n,时间复杂度太高,会超时
if(n%i==0){
len++;
}
while(n%i==0){
n=n/i;
nd[len]++;
}
}
int sig=1;
for(int i=1;i<=len;i++){
sig=sig*(nd[i]+1);
}
if(n>1){//因为循环是到sqrt(n),如果该数比1大,还有一个因数为他本身
sig=sig*2;
}
cout<<sig<<endl;
}
int main(int argc, char** argv) {
int n;
while(cin>>n){
for(int i=0;i<n;i++){
cin>>num[i];
}
for(int i=0;i<n;i++){
cal(num[i]);
}
}
return 0;
}
AC代码(方法2):
#include <iostream>
using namespace std;
int main(int argc, char** argv) {
int div;//约数的个数
int n;
int tmp;
while(cin>>n){
for(int i=0;i<n;i++){
cin>>tmp;
div=1;//,对于大于1的整数至少一个因子
for(int j=2;j*j<=tmp;j++){//sqrt可以避免统计重复的因子
if(tmp%j==0){
div+=2;//为什么加2,因为s=a*b,s%a==0,a,b都为因子
}
}
if(tmp>1){
div+=1;
}
cout<<div<<endl;
}
}
return 0;
}