sgu 441. Set Division

此题明显是求第二类斯特林数,但是由于n 很大 ,k 比较小 , 由于斯特林数是递推的关系,所以可以考虑 矩阵快速幂。

s(n, k ) =  s( n -1 , k-1 ) + k * ( n -1 , k)

行向量是  {s( 1,1 ) , s(1,2) ,s(1,3)......,s(1,k)}

快速幂的矩阵(k * k)  如下格式(6*6)

【1 1 0 0 0 0 】

【0 2 1 0 0 0 】

【0 0 3 1 0 0 】

【0 0 0 4 1 0 】

【0 0 0 0 5 1】

【0 0 0 0 0 6】


快速幂之后的 第一行 第 k列的结果就是答案。

#include<cstdio>
#include<cstring>
#include<iostream>
using namespace std;
#define mod 2007
struct matrix
{
    long long value[11][11];
    int x;
    int y;
    void zero()
    {
        for(int q=1;q<11;q++)
            for(int w=1;w<11;w++)
                value[q][w]=0;
    }
    void one()
    {
        for(int q=0;q<11;q++)
            value[q][q]=1;
    }
    void set(int t){
        for(int i =1;i<=t;i++){
            value[i][i] = i;
        }
        for(int i=2;i<=t;i++){
            value[i-1][i] =1;
        }
    }
    matrix operator*(matrix m1)
    {
        matrix h;
        h.zero();
        for(int q=1;q<11;q++)
            for(int w=1;w<11;w++)
                for(int e=1;e<11;e++)
                {
                    h.value[q][w]+=value[q][e]*m1.value[e][w];
                    h.value[q][w]%=mod;
                }
        return h;
    }
}mm,ma;
matrix pow(matrix m,long long a)
    {
        matrix h;
        if(a==0)
        {
            h.one();
            return h;
        }
        if(a==1)
            return m;
        if((a&1)==0)
            return pow(m*m,a>>1);
        else
            return pow(m*m,a>>1)*m;
    }
int main(){
   int n, k ;
   scanf("%d%d",&n,&k);
   mm.zero();
   mm.set(k);
   mm = pow(mm,n-1);
   cout << mm.value[1][k]<<endl;
   
    return 0;
}





你可能感兴趣的:(sgu 441. Set Division)