<<  < 2013 - >  >>
1 2 3 4 5 6 7
8 9 10 11 12 13 14
15 16 17 18 19 20 21
22 23 24 25 26 27 28
29 30




昨天做了一个挺有意思的数论题目,感觉不错。今天在此写个题解。
题目为POJ1150
题目描述是:

Description

In this problem you will be given two decimal integer number N, M. You will have to find the last non-zero digit of the NPM.This means no of permutations of N things taking M at a time.

Input

The input contains several lines of input. Each line of the input file contains two integers N (0 <= N<= 20000000), M (0 <= M <= N).

Output

For each line of the input you should output a single digit, which is the last non-zero digit of NPM. For example, if NPM is 720 then the last non-zero digit is 2. So in this case your output should be 2.

Sample Input

10 10
10 5
25 6

Sample Output

8
4
2
题意就是给你n和m求P(n,m)的最末尾非0位。
这是一个比较典型的数位分解的题目。
首先其组合数实际上就是求阶乘。
因为把每个数都因数分解成质数后,只有2*5才末尾才会为0.
这样把每个抵消后就可以把所有的0消除掉了。
同时,还有一个地方就是对于任意整数n*10+x和x对一个数的积末尾数是一样的(因为末尾已经不存在0)。什么意思呢?举个例子:19*7的积和19*37的积个位是一样的,与9*7的积各位也是一样的。
这样问题有转化为在求出在阶乘中总共有多少个以某个数字结尾的数的个数。这些数字包括2,3,5,7,9。把这些求出来就可以搞定了。
具体的实现过程中还是有好多细节,好多坑的。

上我的代码:(部分参考)
#i nclude <iostream>
#i nclude <cstdio>
#i nclude <cstring>
using namespace std;

int num2,num3,num5,num7,num9,n,m;
int tab[4][4]=
{
    6,2,4,8,
    1,3,9,7,
    1,7,9,3,
    1,9,1,9
};
int get2(int x)
{
    if (x==0) return 0;
    return x/2+get2(x/2);
}
int get5(int x)
{
    if (x==0) return 0;
    return x/5+get5(x/5);
}
int g(int x,int y)
{
    if (x==0) return 0;
    return x/10+(x%10>=y)+g(x/5,y);
}
int get(int x,int y)
{
    if (x==0) return 0;
    return get(x/2,y)+g(x,y);
}

int main()
{
    while (scanf("%d%d",&n,&m)!=EOF)
    {
        num2=get2(n)-get2(n-m);
        num5=get5(n)-get5(n-m);
        num3=get(n,3)-get(n-m,3);
        num5=get(n,5)-get(n-m,5);
        num7=get(n,7)-get(n-m,7);
        num9=get(n,9)-get(n-m,9);
        if (num2<num5)
        {
            puts("5");
            continue;
        }
        num2-=num5;
        int ans=1;
        if (num2>0) ans*=tab[0][num2%4];
        ans*=tab[1][num3%4];
        ans*=tab[2][num7%4];
        ans*=tab[3][num9%4];
        printf("%d\n",ans%10);
    }
    return 0;
}
这是个好题,也是数学题中很有代表性的一个题目。
好好领悟,总会有收获的。
可悲的是:题目是我看过大神的题解后才会做的。T_T

  • 标签:数论 
  • love999888我还是小学生 看不明白
    love258258你是高年级的吗
    love258258
    acm_letgo好像不知道怎么回复评论
    xingxingjiyi感觉像是c
    发表评论:
    天涯博客欢迎您!