用C语言输入一个自然数n,求n!,同时统计结果中有多少个0

如题所述

不用开新问题了,我已经把原回答修改了,改成你要的字符串运算。

但即使这样,n也不要太大,看我测试数据就知道了。

代码原理:n!就是循环累计乘法,多位数字符串与多位数字符串相乘和人算法一样,就是其中一个字符串每一位数字和另一个字符串数字相乘,同时所有乘积移位累加。

注意:我写的所有字符串运算函数,没有写字符串验证,如果你想单独把函数拿出来用,记得写个输入验证,不要把非数字的字符串传进去。

#include <stdio.h>
#include <string.h>
#include <malloc.h>
#include <conio.h>
void meError(void *p);//内存申请失败
char *addByStr(char *str1,char *str2);
char *inversion(char *str);//倒置字符串
char *multByStr1(char *str1,char c2);//多位数字符串与单位数字符串相乘
char *multByStr2(char *str1,char *str2);//多位数字符串相乘
char *p10ByStr(char *str,int n);//字符串数字乘n个10
char *num2Str(int n);//数字转字符串
int main()
{
    int n,i,len,cnt=0;
    char *nStr=NULL;
    while(1)
    {
        nStr=(char *)malloc(sizeof(char)*2);
        meError(nStr);
        nStr[0]='1',nStr[1]=0;
        printf("输入一个自然数n,求n!\n");
        scanf("%d",&n);
        for(i=1;i<=n;i++)
            nStr=multByStr2(nStr,num2Str(i));
        printf("计算结果:%s\n",nStr);
        len=strlen(nStr);
        for(i=len-1;i>=0;i--)
            if(nStr[i]=='0')
                cnt++;
        printf("结果包含%d个0\n\n",cnt);
        free(nStr);
        nStr=NULL;
    }
    return 0;
}
char *num2Str(int n)//数字转字符串
{
    int i=0,len=1;
    char *str=NULL,*strSave=NULL;
    while(n)
    {
        if(str==NULL)
        {
            str=(char *)malloc(sizeof(char)*2);
            meError(str);
        }
        else
        {
            strSave=(char *)realloc(str,sizeof(char)*(len+1));
            meError(strSave);
            str=strSave;
            strSave=NULL;
        }
        str[i]=n%10+'0';
        str[i+1]=0;
        i++;
        len++;
        n=n/10;
    }
    inversion(str);
    return str;
}
char *p10ByStr(char *str,int n)//字符串数字乘n个10,注意:str必须是动态申请内存!!
{
    int len=strlen(str),i;
    char *p=NULL,*strSave=NULL;
    if(n>0)
    {
        strSave=realloc(str,sizeof(char)*(len+1+n));
        meError(strSave);
        str=strSave;
        p=&str[len];
        for(i=0;i<n;i++)
            *p='0',p++;
        *p=0;
    }
    return str;
}
char *multByStr2(char *str1,char *str2)//多位数字符串相乘
{
    int len2=strlen(str2),i,j=0;
    char **addStrs=(char **)malloc(sizeof(char *)*len2),*sum0=NULL,*sum1=NULL,*sum=NULL,c2;
    meError(addStrs);
    for(i=len2-1;i>=0;i--)
    {
        c2=str2[i];
        addStrs[j++]=multByStr1(str1,c2);//这里addStrs存储的是str1和str2每一位的乘积
    }
    //--------sum0和sum1交替,为了及时释放内存-------
    sum0=(char *)malloc(sizeof(char)*2);
    meError(sum0);
    sum0[0]='0',sum0[1]=0;;
    for(i=0;i<len2;i++)
    {
        addStrs[i]=p10ByStr(addStrs[i],i);//在乘法运算中,最后累加要乘10
        if(sum1==NULL)
        {
            sum1=addByStr(sum0,addStrs[i]);
            free(sum0);
            sum0=NULL;
        }
        else
        {
            sum0=addByStr(sum1,addStrs[i]);
            free(sum1);
            sum1=NULL;
        }
        free(addStrs[i]);
        addStrs[i]=NULL;
    }
    if(sum0)
        sum=sum0;
    else
        sum=sum1;
    free(addStrs);
    addStrs=NULL;
    return sum;
}
char *multByStr1(char *str1,char c2)//多位数字符串与单位数字符串相乘
{
    int len1=strlen(str1),i=len1-1,a,b,c=0;
    char *mulStr=(char *)malloc(sizeof(char)*(len1+2)),*p=mulStr;
    meError(mulStr);
    memset(mulStr,0,sizeof(char)*(len1+2));
    b=c2-'0';
    while(1)
    {
        a=str1[i]-'0';
        *p=((a*b)+c)%10+'0';
        c=((a*b)+c)/10;
        p++;
        if(i==0)
        {
            if(c>0)
                *p=c+'0';
            break;
        }
        i--;
    }
    inversion(mulStr);
    return mulStr;
}
char *addByStr(char *str1,char *str2)
{
    int len1=strlen(str1),len2=strlen(str2),maxSize,i=len1-1,j=len2-1,a,b,c=0;
    char *addStr=NULL,*p=NULL;
    if(len1>len2)//多留两位,一位给结束符号,一位给进位
        maxSize=len1+2;
    else
        maxSize=len2+2;
    addStr=(char *)malloc(sizeof(char)*maxSize);
    meError(addStr);
    memset(addStr,0,sizeof(char)*maxSize);
    p=addStr;
    while(1)
    {
        if(i<0)
            a=0;
        else
            a=str1[i]-'0';
        if(j<0)
            b=0;
        else
            b=str2[j]-'0';
        *p=(a+b+c)%10+'0';//从后往前,每一位做加运算并保存余数和进位(数组中结果是反向存储的,最后再将数组倒置)
        c=(a+b+c)/10;
        p++;
        if(i<=0 && j<=0)
        {
            if(c>0)
                *p=c+'0';
            break;
        }
        i--;
        j--;
    }
    //--------------数组倒置------------------
    inversion(addStr);
    return addStr;
}
char *inversion(char *str)//倒置字符串
{
    char *p=str,*pd=&str[strlen(str)-1],cs;
    while(p<pd)
    {
        cs=*p;
        *p=*pd;
        *pd=cs;
        p++;
        pd--;
    }
    return str;
}
void meError(void *p)//内存申请失败
{
    if(p==NULL)
    {
        printf("\n异常:内存申请失败!回车结束程序!\n");
        while(getch()!='\r');
        exit(0);
    }
}

温馨提示:内容为网友见解,仅供参考
第1个回答  2008-01-07
n!应该会求吧.就是1*2*3*4*5*....*n.
1.把数字转化成字符串,统计结果中0的个数.

#include <stdio.h>
#include <math.h>

int Digit(int n)
{
int d = 0;
do{
++d;
n /= 10;
}while(n > 0 || n < 0);

return d;
}

void itoa(int n, char* des)
{
if(n > 0)
{
int x = (int)pow(10.0, Digit(n)-1);
*des = n / x + '0';
itoa(n-(n/x)*x, des+1);
}
else{
*des = '\0';
}

}

void itora(int n, char* des)
{
*des++ = n > 0 ? itora(n/10, des), n % 10 + '0' : '\0';
}

int main()
{
char b[10], c[10];
itoa(12345, b);
itora(12345, c);
printf("%s %s", b, c);
}

2.用除法,看余数.
除以10没有余数,说明个位是0.然后把个位变成0,除以100,没有余数,说明十位是0.然后把十位再变成0,除以1000,没有余数,说明百位是0......
自己写程序吧.

参考资料:http://zhidao.baidu.com/question/25887067.html?si=1

本回答被提问者采纳
第2个回答  2008-01-07
看看我这个吧我调试运行过的没问题
完全满足你的要求
#include<stdio.h>
main()
{
int n,i;
int sum=1,zero=0;
printf("请输入一个自然数:");
scanf("%d",&n);
for(i=1;i<=n;i++)
sum=sum*i;
printf("n!=%d\n",sum);
while(sum!=0)
{
if( sum%10 == 0 )
zero++;
sum=sum/10;
}
printf("结果中有%d个0\n",zero);

}

你认为好的话采我吧
第3个回答  2008-01-08
/*数不能太大了,不然就溢出了。如果想求数特别大的情况,把悬赏提高180分就帮你,还要注明最高多少位的。阶乘的运算,结果太大了。*/
#include<stdio.h>

void main()
{
long n=1,m;

printf("Input the n please!\n");
scanf("%d",&m);
if((m<13)&&(m>0))
{
for(;m>0;m--)
n*=m;
printf("The answer is %d\n",n);
m=0;
while((n/10)!=0)
{
if(((n/10)*10)==n)
m++;
n/=10;
}
printf("There are %d '0' in the answer!\n",m);
}
else
printf("The number will overflow!");
}
第4个回答  2008-01-07
以上方法太复杂,最简单的是:
#include<stdio.h>
main()
{
int n,i;
int sum=1,zero=0;
printf("请输入一个自然数:");
scanf("%d",&n);
for(i=1;i<=n;i++)
sum=sum*i;
while(n/=5)
{
zero += n;
}
printf("n!=%d\n",sum);
printf("结果中有%d个0\n",zero);
}

用C语言输入一个自然数n,求n!,同时统计结果中有多少个0
nStr[0]='1',nStr[1]=0; printf("输入一个自然数n,求n!\\n");

输入一个自然数n,求n!,同时统计结果中有多少个0程序分析解释
scanf("%d",&n); \/*输入一个整数N*\/ i=fun (n); \/*求N!未尾0的个数*\/ k=jc (n); \/*求N!*\/ printf("%d! is %10.0f\\nthere are %d zero!\\n",n,k,i); \/*输出结果*\/ } int fun(int x)\/*定义fun函数*\/ { int i=0;while(x!=0)\/*请参照求2008!未尾0的个数...

输入一个自然数n,求n!,同时统计结果中有多少个0.
调用个统计结果中0的个数的函数int count(int m){ int i=0; int k; while(m) { k=m%10; if(k==0) i++; \/\/统计阶乘结果中,0的个数。 m=m\/10; } return i;} 主函数需要输入自然数n和调用两个函数的东西我就不写了。希望你能看下这个过程...

用c语言实现,输入一个自然数n(n>0),统计n中有多少个0?
\/\/ 103045,每次除10:103045、10304、1030、103、10、1 \/\/中1030和10能被10整除,那么就有两个零。void Func( int n){ int num = 0;int local_n = n;if(local_n <0){ printf("Error, the parameter should not < 0\\n");} while(local_n >0){ if((local_n % 10)== 0)...

输入一个自然数n,求n!,同时统计结果中有多少个n?用c语言编写
楼上的你写得太复杂了 我估计楼主也是刚学C的吧 很难看懂你的 哈哈楼主我的简单看看我的吧 include<stdio.h> main(){ int i,n=9,sum=1,count=0; \/\/自己改n的值吧 n大于9就复杂点了 先看懂这个程序吧 for(i=1;i<=n;i++)sum=sum*i;printf("n!=%d\\n",sum);while(sum>0){ i...

用C语言编写程序。输入一个自然数N,输出N的位数。
要输出一个自然数N的位数,可以使用log10函数。log10函数的返回值是以10为底的对数,可以用来计算一个数的位数。具体实现如下:include <stdio.h>#include <math.h>int main() {int n;scanf("%d", &n);int digits = (int)log10(n) + 1;printf("%d\\n", digits);return 0;} 首先,...

用C语言编写程序从键盘输入一个正整数数n,输出n!
编程实现计算n!,也就是n*(n-1)*(n-2)*...*1,可以通过递归函数来实现,具体实现方法:include <stdio.h>int Func(int n){if(n < 2)return 1;else return n*Func(n-1);} void main(){int n =0,m=0;printf("请输入一个正整数:")scanf("%d",&n);m=Func(n);printf("...

C语言 自守数。任意输入一个自然数,判断是否自守数并输出
include <string.h> int main(void) { int n, square, len;char str[20];printf("请输入一个自然数:");scanf("%d", &n);square = n * n;sprintf(str, "%d", square);len = strlen(str);if (str[len-1] == '0' + n) { printf("%d yes %d*%d=%d\\n", n, n, n, ...

用C语言编写程序从键盘输入一个正整数数n,输出n,是什么情况?
C语言实现如下: #include<stdio.h> void main() { int i,j,k,n; scanf("%d",&n); \/\/输入一个正整数n for(i=2;i<=n;i++) { k=0; for(j=1;j<=i-1;j++) if(i%j==0) k+=j; if(k==i) printf("%d\\n",i); \/\/判断i的所有真因子之和是否等于i,是则输出i。 } } 注意:建...

读入一个自然数n,计算其各位数字之和,并用英文输出和的每一位数字。 C...
include <stdio.h>void main(){int i=0,j,n,t,s=0,a[100];char p[][6]={"zero","one","two","three","four","five","six","seven","eight","nine"};scanf("%d",&n);while(n){a[i]=n%10;s+=a[i],n\/=10;i++;}for(j=0;...

相似回答