问题描述:

林记在做数学习题的时候,经常遇到这种情况:苦思冥想了很久终于把问题解出来,结果发现答案是0,久而久之林记在得到习题答案是0的时候就没有了做出一道难题的成就感。于是林记决定:以后出题,答案一定不能是0,例如求n!最低位非零数这样的习题就很不错了。

现在林记提出了一个更难一点的问题:求n!在K进制下的最低位非零数。其中K符合一些特殊的条件:K是由若干个互不相同的质数相乘得出来的,例如K=2,3,5,6,7,10……

输入格式:

首先输入的第一行是一个整数Q,表示询问的个数。

接下来是Q个询问,每个询问有两行组成,第一行首先是一个整数nk,然后紧跟着nk个正整数:m1,m2,……mnk,则K为所有mi的乘积,输入保证mi是有若干个互不相同的质数相乘得出来的。接下来一行给出一个整数n。

输出格式:

对于每个询问,如果K不符合题目限制,则输出“I hate zero.”(不用加双引号)。否则输出K进制下n!的最低位非零数。

输入样例:

输出样例:

3

1 17

9

1 6

49

1 93

16

15

2

78

数据范围:

对于10%的数据满足:n≤10,K≤100

对于50%的数据满足:n≤100000,K≤100

对于100%的数据满足:n≤1015,K≤1015,mi≤106,Q≤20

【分析】

  快速求阶乘大法!!

  下面某步还用到 威尔逊定理 (p-1)!%p=p-1...... (也可以预处理出来的)

  把m分解质因数,对于质数p单独求n!=p^a+b的a的值和b%p的值(logn递归搞定),然后用中国剩余定理合并起来。

  

  long long的乘积会爆,所以...要用那个不断乘2的方法:

  

LL mul(LL x,LL y,LL p)
{
if(x==0||y==0) return 0;
LL now=x,yy=y,add=0;
while(yy>1)
{
if(yy%2!=0) add=(add+now)%p;
now=(now*2)%p;
yy/=2;
}
return (now+add)%p;
}

代码如下:

 #include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<queue>
#include<cmath>
using namespace std;
#define Maxn 1000010
#define INF 0xfffffff
#define LL long long LL nk,m[Maxn],ml,K;
LL n;
LL prime[Maxn],pl;
bool q[Maxn];
LL mn; LL mymin(LL x,LL y) {return x<y?x:y;} void get_pri(LL mx)
{
pl=;
memset(q,,sizeof(q));
for(LL i=;i<=mx;i++)
{
if(q[i]) prime[++pl]=i;
for(int j=;j<=pl;j++)
{
if(i*prime[j]>mx) break;
q[i*prime[j]]=;
if(i%prime[j]==) break;
}
}
} bool div(LL mm)
{
for(LL i=;prime[i]*prime[i]<=mm;i++) if(mm%prime[i]==)
{
m[++ml]=prime[i];
mm/=prime[i];
while(mm%prime[i]==) return ;
}
if(mm!=) m[++ml]=mm;
return ;
} bool init()
{
ml=;
scanf("%lld",&nk);
K=;
bool ok=;
for(int i=;i<=nk;i++)
{
LL mm;
scanf("%lld",&mm);K*=mm;
if(!div(mm)) ok=;
}
sort(m+,m++ml);
for(LL i=;i<=ml;i++) if(m[i]==m[i-]) {ok=;break;}
scanf("%lld",&n);
if(!ok) return ;
return ;
} LL f[Maxn]; struct node{LL x,y;};
node c[Maxn]; node ffind(LL x,LL p)
{
node ans;
if(x<p)
{
ans.x=;
ans.y=f[x];
return ans;
}
ans.x=;ans.y=;
node t=ffind(x/p,p);
ans.x+=t.x; ans.y*=t.y;
ans.y=(ans.y*f[x%p])%p;
ans.x+=x/p; LL y=(((x/p)%)==)?:p-;
ans.y=(ans.y*y)%p; return ans;
} LL mul(LL x,LL y,LL p)
{
if(x==||y==) return ;
LL now=x,yy=y,add=;
while(yy>)
{
if(yy%!=) add=(add+now)%p;
now=(now*)%p;
yy/=;
}
return (now+add)%p;
} LL qpow(LL x,LL b,LL p)
{
LL ans=;
while(b)
{
if(b&) ans=mul(ans,x,p);
x=mul(x,x,p);
b>>=;
}
return ans;
} LL d[Maxn]; LL get_ans()
{
mn=c[].x;
for(LL i=;i<=ml;i++) mn=mymin(mn,c[i].x);
for(LL i=;i<=ml;i++)
{
if(c[i].x>mn) {d[i]=;continue;}
LL bm=qpow(K/m[i],mn,m[i]);
bm=qpow(bm,m[i]-,m[i]);
d[i]=mul(bm,c[i].y,m[i]);//(bm*c[i].y)%m[i];
}
LL ans=;
for(LL i=;i<=ml;i++)
{
LL y=qpow(K/m[i],m[i]-,m[i]);
//ans=(ans+((K/m[i]*y)%K)*d[i])%K;
ans=(ans+mul(mul(K/m[i],y,K),d[i],K))%K;
}
return ans;
} int main()
{
get_pri();
int T;
scanf("%d",&T);
while(T--)
{
if(!init()) {printf("I hate zero.\n");continue;}
for(LL i=;i<=ml;i++)
{
f[]=;
for(LL j=;j<m[i];j++) f[j]=(f[j-]*j)%m[i];
c[i]=ffind(n,(LL)m[i]);
}
printf("%lld\n",get_ans());
}
return ;
}

2016-09-04 19:37:26

【GDOI 2011 DAY2 T3】零什么的最讨厌了 (快速求阶乘、中国剩余定理)的更多相关文章

  1. 【NOIP 2017】Day2 T3 列队

    Problem Description \(Sylvia\) 是一个热爱学习的女孩子. 前段时间,\(Sylvia\) 参加了学校的军训.众所周知,军训的时候需要站方阵. \(Sylvia\) 所在的 ...

  2. 数论-质数 poj2689,阶乘分解,求阶乘的尾零hdu1124, 求尾零为x的最小阶乘

    /* 要求出[1,R]之间的质数会超时,但是要判断[L,R]之间的数是否是素数却不用筛到R 因为要一个合数n的最大质因子不会超过sqrt(n) 所以只要将[2,sqrt(R)]之间的素数筛出来,再用这 ...

  3. 【BZOJ 4518】【SDOI 2016 Round1 Day2 T3】征途

    比较明显的斜率优化DP,省选时因为时间太紧张和斜率DP写得不熟等原因只写了60分的暴力DP,其实当时完全可以对拍来检验标算的正确,但是我当时too naive- 很快打完了,调了将近一晚上QAQ,因为 ...

  4. 【NOIP 2013 DAY2 T3】 华容道(spfa)

    题目描述 [问题描述] 小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果能完成, 最少需要多少时间. 小 ...

  5. 【NOIP 2015 DAY2 T3】 运输计划 (树链剖分-LCA)

    题目背景 公元 2044 年,人类进入了宇宙纪元. 题目描述 L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之间,这 n-1 条航道连通了 L 国的所有星球. 小 P 掌管一家 ...

  6. 【NOIP2015提高组】 Day2 T3 运输计划

    题目描述 L 国有 n 个星球,还有 n-1 条双向航道,每条航道建立在两个星球之间,这 n-1 条航道连通了 L 国的所有星球. 小 P 掌管一家物流公司,该公司有很多个运输计划,每个运输计划形如: ...

  7. 一道搜索题【2013 noip提高组 DAY2 t3】华容道

    这篇不多说,具体的解释都在程序里 题目描述 [问题描述] 小 B 最近迷上了华容道,可是他总是要花很长的时间才能完成一次.于是,他想到用编程来完成华容道:给定一种局面, 华容道是否根本就无法完成,如果 ...

  8. 【NOIP 2015】Day2 T3 运输计划

    Problem Background 公元 \(2044\) 年,人类进入了宇宙纪元. Description 公元\(2044\) 年,人类进入了宇宙纪元. $L $国有 \(n\) 个星球,还有 ...

  9. 【NOIP 2016】Day2 T3 愤怒的小鸟

    Problem Description \(Kiana\) 最近沉迷于一款神奇的游戏无法自拔. 简单来说,这款游戏是在一个平面上进行的. 有一架弹弓位于 \((0,0)\) 处,每次 \(Kiana\ ...

随机推荐

  1. ERROR 1114 (HY000): The table 'adv_date_tmp' is full(Mysql临时表应用)

    场景:需要对现在数据库的数据进行批量的进行is_del=1的操作,但是遇到一个问题,在执行sql的时候发现sql不能在查询特定表的时候再嵌套查询来做update的操作,经过讨论,后续我们想到用临时表的 ...

  2. Qt Quick 简单教程 - 1 (代码备忘)

    qmlscene 未安装 由于出现上面的情况,我开始转战Windows 下学习,昨天安装好了Qt Sdk了,哟吼吼吼. mail.qml内容: import QtQuick 2.3 import Qt ...

  3. 关于ContentProvider的批量操作

    今天看公司代码,发现在批量插入通话记录和联系人的时候,用了一个 ArrayList<ContentProviderOperation> ops = new ArrayList<Con ...

  4. 响应式框架中,table表头自动换行的解决办法

    最近在用bootstrap开发网站,在处理一张table的时候发现,通过PC端查看样式正常,在手机上查看时,因为屏幕小,表格被压缩的厉害,表头和数据变形如下图 后来网上找了一下,发现一个好用的CSS属 ...

  5. 10个你可能不知道的JavaScript小技巧

    1.变量转换 看起来很简单,但据我所看到的,使用构造函数,像Array()或者Number()来进行变量转换是常用的做法.始终使用原始数据类型(有时也称为字面量)来转换变量,这种没有任何额外的影响的做 ...

  6. ORA-12011+ORA-06512–job执行失败问题

    oracle库中的一个job,正常运行了一年多,因某种原因导致无法运行,系统尝试多次均失败之后下次执行时间就变成了 4000/1/1. 现导致job失败的问题已经解决,从新运行job时报 12011和 ...

  7. JavaSE生成随机数

    今天呢,老师讲了一下怎么用jvm实现随机数(本人对此很感兴趣),一个简单的随机100以内整数的代码如下: /** 生成随机数 */ import java.util.Random; public cl ...

  8. requirejs实践一 加载JavaScript文件

    首先,目录如下(根目录有一个index.html文件.有一个scripts文件夹): scripts文件夹有如下的文件 以下是index.html代码 <!DOCTYPE html> &l ...

  9. PAT_1046 划拳

    啦啦啦.今天晚上火车回学校了.= =还是挺舍不得家里的. 题目描述: 划拳是古老中国酒文化的一个有趣的组成部分.酒桌上两人划拳的方法为:每人口中喊出一个数字,同时用手比划出一个数字.如果谁比划出的数字 ...

  10. IOS 学习笔记 2015-04-03 OC-API-文件读写

    // // WPFileHelper.m // OC-API-文件操作 // // Created by wangtouwang on 15/4/3. // Copyright (c) 2015年 w ...