POJ 1150 The Last Non-zero Digit 数论+容斥
POJ 1150 The Last Non-zero Digit 数论+容斥
题目地址: id=1150" rel="nofollow" style="color:rgb(0,136,204);text-decoration:none;">POJ 1150
题意:
求排列P(n, m)后面第一个非0的数。
分析:
为了熟悉题目中的理论。我先做了俩0基础的题目: id=1401" rel="nofollow" style="color:rgb(0,136,204);text-decoration:none;">POJ 1401 pid=954" rel="nofollow" style="color:rgb(0,136,204);text-decoration:none;">NYOJ 954
1401 && ZOJ 2202 Factorial 阶乘N!的末尾零的个数
954 求N!二进制末尾几个0
这题想了一下。十进制末尾几个0能够转化为几个5因子。二进制最后一位非0能够转化为2因子。可是10进制就不行了,并且这个不是单单N!。而是n!/m!。orz。
实在太弱仅仅能研究题解,推荐SCAU_Lyon巨巨的题解和abilitytao巨巨的题解。
然后我坐在电脑前面看了一晚上题解,最终。我发现我太弱了,我好不easy有点看懂了。
大概讲一下吧。
我们要把排列转化成前面两题的形式。
P(n, m) = n!/(n-m)!,我们让m = n - m直接按n!/m!做,也就是求m∗(m+1)∗(m+2)∗....∗n了。
这时候,我们仅仅要统计在[m, n]这个范围里面的数的最后一位即可了,假设直接去暴力会跪。由于我们不仅要统计每一个数的最后一位,另一些2和5因子混在数中,我们还要消掉那些因子,然后取末位。
于是要把2和5抽出来,然后就仅仅剩3,7,9,统计抽完后的3,7,9,然后还得记得:由于这里面2可能会比5多,所以要把多出来的2算出来。
仅仅要知道怎样统计n!,就能统计排列。
个中计算有非常厉害的技巧,表示orz。
一、计算n!中消除2,5后末位为x的公式为:
f(n, x) = n/10 + (n%10>=x) + f(n/2, x) + f(n/2, 5) - f(n/10, x).
解释下:
1. x限定3,7,9。
2. n/10 + (n%10>=x)也就是:每十个数以内末数字是3,7,9在没有除去2和5两种因子前都仅仅会出现一次。
3. 加上抽出2和5后的子问题。(这里跟前面两题原理一样)
4. 抽出2和5的时候,会多抽出了一次10,这时候就要用容斥定理减去。
二、然后计算多余的2就比較easy理解了。就跟前面俩题一样。求出2的个数和5的个数。减一下就是n!中多出来的2的个数了。
三、然后我们就能得到[m,n]中抽出2,5后末位为3,7,9的个数,以及多出来的2的个数了。
这时候假设直接while(cnt--)去算可能会超时+爆范围。
我们能够发现2^n,3^n,7^n,9^n的末位都是有规律可寻的。于是就能直接算了。
具体细节见代码。
代码:
/*
* Author: illuz <iilluzen[at]gmail.com>
* File: 1150.cpp
* Create Date: 2014-05-26 22:28:45
* Descripton:
*/
#include <cstdio>
#include <cstring>
const int N = 2e5;
int cnt3, cnt7, cnt9, cnt2;
int n, m;
int rec[10][N];
// 计算n!中消除2,5后末位为x的数量
int f(int n, int k) {
if (n < 1)
return 0;
if (n < N && rec[k][n] != -1)
return rec[k][n];
int ret = n / 10 + (n % 10 >= k) + f(n / 2, k) + f(n / 5, k) - f(n / 10, k);
if (n < N)
rec[k][n] = ret;
return ret;
}
// 多出来的2的个数
int more(int n) {
int num2 = 0, num5 = 0;
int t = n;
while (t != 0) {
num2 += t / 2;
t /= 2;
}
while (n != 0) {
num5 += n / 5;
n /= 5;
}
return num2 - num5;
}
int main()
{
memset(rec, -1, sizeof(rec));
while (~scanf("%d%d", &n, &m)) {
if (m == 0) {
puts("1");
continue;
}
m = n - m;
cnt2 = more(n) - more(m);
cnt3 = f(n, 3) - f(m, 3);
cnt7 = f(n, 7) - f(m, 7);
cnt9 = f(n, 9) - f(m, 9);
// printf("%d %d %d %d\n", cnt2, cnt3, cnt7, cnt9);
// 2 4 8 6
if (cnt2-- == 0)
cnt2 = 1;
else if (cnt2 % 4 == 0)
cnt2 = 2;
else if (cnt2 % 4 == 1)
cnt2 = 4;
else if (cnt2 % 4 == 2)
cnt2 = 8;
else
cnt2 = 6;
// 1 3 9 7
if (cnt3 % 4 == 0)
cnt3 = 1;
else if (cnt3 % 4 == 1)
cnt3 = 3;
else if (cnt3 % 4 == 2) cnt3 = 9;
else
cnt3 = 7;
// 1 7 9 3
if (cnt7 % 4 == 0)
cnt7 = 1;
else if (cnt7 % 4 == 1)
cnt7 = 7;
else if (cnt7 % 4 == 2)
cnt7 = 9;
else
cnt7 = 3;
// 1 9
if (cnt9 % 2 == 0)
cnt9 = 1;
else
cnt9 = 9;
printf("%d\n", cnt2 * cnt3 * cnt7 * cnt9 % 10);
}
return 0;
}
POJ 1150 The Last Non-zero Digit 数论+容斥的更多相关文章
- Coprime (单色三角形+莫比乌斯反演(数论容斥))
这道题,先说一下单色三角形吧,推荐一篇noip的论文<国家集训队2003论文集许智磊> 链接:https://wenku.baidu.com/view/e87725c52cc58bd631 ...
- hdu1695:数论+容斥
题目大意: 求x属于[1,b]和 y属于[1,d]的 gcd(x,y)=k 的方案数 题解: 观察发现 gcd()=k 不好处理,想到将x=x/k,y=y/k 后 gcd(x,y)=1.. 即问题转化 ...
- CROC 2016 - Elimination Round (Rated Unofficial Edition) F - Cowslip Collections 数论 + 容斥
F - Cowslip Collections http://codeforces.com/blog/entry/43868 这个题解讲的很好... #include<bits/stdc++.h ...
- 数论 + 容斥 - HDU 4059 The Boss on Mars
The Boss on Mars Problem's Link Mean: 给定一个整数n,求1~n中所有与n互质的数的四次方的和.(1<=n<=1e8) analyse: 看似简单,倘若 ...
- 数论 + 容斥 - HDU 1695 GCD
problem's Link mean 给定五个数a,b,c,d,k,从1~a中选一个数x,1~b中选一个数y,使得gcd(x,y)=k. 求满足条件的pair(x,y)数. analyse 由于b, ...
- HDU - 2204 Eddy's爱好 (数论+容斥)
题意:求\(1 - N(1\le N \le 1e18)\)中,能表示成\(M^k(M>0,k>1)\)的数的个数 分析:正整数p可以表示成\(p = m^k = m^{r*k'}\)的形 ...
- 【bzoj2393】Cirno的完美算数教室 数论容斥
Description ~Cirno发现了一种baka数,这种数呢~只含有2和⑨两种数字~~ 现在Cirno想知道~一个区间中~~有多少个数能被baka数整除~ 但是Cirno这么天才的妖精才不屑去数 ...
- HDU 5514 Frogs (数论容斥)
题意:有n只青蛙,m个石头(围成圆圈).第i只青蛙每次只能条ai个石头,问最后所有青蛙跳过的石头的下标总和是多少? 析:首先可以知道的是第 i 只青蛙可以跳到 k * gcd(ai, m),然后我就计 ...
- UOJ #129 / BZOJ 4197 / 洛谷 P2150 - [NOI2015]寿司晚宴 (状压dp+数论+容斥)
题面传送门 题意: 你有一个集合 \(S={2,3,\dots,n}\) 你要选择两个集合 \(A\) 和 \(B\),满足: \(A \subseteq S\),\(B \subseteq S\), ...
随机推荐
- 机器学习中常用的距离及其python实现
1 概述 两个向量之间的距离(此时向量作为n维坐标系中的点)计算,在数学上称为向量的距离(Distance),也称为样本之间的相似性度量(Similarity Measurement).它反映为某类事 ...
- ElasticSearch入门介绍之安装部署(二)
散仙,在上篇文章对ElasticSearch整体入门作了个介绍,那么本篇我们来看下,如何安装,部署es,以及如何安装es的几个比较常用的插件. es的安装和部署,是非常简单方便的,至少这一点散仙在es ...
- 解决IDEA中,maven依赖不自动补全的问题
转载: 作者:七个榴莲链接:https://www.jianshu.com/p/46a423bdde31来源:简书 遇到的问题:Maven依赖不自动补全 在idea上使用maven插件时,发现在pom ...
- python基础-迭代器
1.迭代:指的是一个重复的过程,每一次重复称为一次迭代,并且每一次重复的结果 是下一次重复的初始值 2.为什么要有迭代器 对于序列类型:str list tuple 可以依赖索引来迭代取值,但是对于d ...
- 【心无旁骛】vuex-typescript-example
我不惮以最大的赞美去赞美这样的项目,真的是非常有创意又有能力. 先放上我喜欢的这个项目的开源地址:https://github.com/gluons/vuex-typescript-example 我 ...
- java虚拟机(十二)--可视化工具分析GC日志
在上篇博客中,我们学习了Parallel.CMS.G1三种垃圾收集器的日志格式,本次我们通过工具去分析日志,会更加的直观 日志格式博客地址:java虚拟机(十一)--GC日志分析 GCeasy: 这是 ...
- HZOI20190821模拟28题解
题面:https://www.cnblogs.com/Juve/articles/11390839.html 所有官方正解在我的文件里 A. 虎 算法1:我们发现非关键边与黑色边去掉以后,答案就是将所 ...
- substring() 方法用于提取字符串中介于两个指定下标之间的字符。
substring() 方法用于提取字符串中介于两个指定下标之间的字符. 语法 stringObject.substring(start,stop) 参数 描述 start 必需.一个非负的整数,规定 ...
- 【JZOJ5179】【NOI2017模拟6.29】哈哈
题意 给定一个长度为n的序列,你可以进行若干次操作: 选择一个区间,删掉,并获得Val[Len]的得分,Len为这个区间的长度: 其中这个区间满足: 1.相邻两个数差的绝对值为1 2.每个数都大于相邻 ...
- Spring AOP(二)--注解方式
本文介绍通过注解@AspectJ实现Spring AOP,这里要重点说明一下这种方式实现时所需的包,因为Aspect是第三方提供的,不包含在spring中,所以不能只导入spring-aop的包,为了 ...