careercup-中等难度 17.3

17.3 写一个算法计算n的阶乘末尾0的个数?

解答:

首先,算出n的阶乘的结果再去计算末尾有多少个0这种方法是不可取的, 因为n的阶乘是一个非常大的数,分分种就会溢出。我们应当去分析, 是什么使n的阶乘结果末尾出现0。

n阶乘末尾的0来自因子5和2相乘,5*2=10。因此,我们只需要计算n的阶乘里, 有多少对5和2。注意到2出现的频率比5多,因此,我们只需要计算有多少个因子5即可。 我们可以列举一些例子,看看需要注意些什么:

5!, 包含1*5, 1个5
10!, 包含1*5,2*5, 2个5
15!, 包含1*5,2*5,3*5, 3个5
20!, 包含1*5,2*5,3*5,4*5, 4个5
25!, 包含1*5,2*5,3*5,4*5,5*5, 6个5

给定一个n,用n除以5,得到的是从1到n中包含1个5的数的个数;然后用n除以5去更新n, 相当于把每一个包含5的数中的因子5取出来一个。然后继续同样的操作,让n除以5, 将得到此时仍包含有5的数的个数,依次类推。最后把计算出来的个数相加即可。 比如计算25的阶乘中末尾有几个0, 先用25除以5得到5,表示我们从5,10,15,20,25中各拿一个因子5出来,总共拿了5个。 更新n=25/5=5,再用n除以5得到1,表示我们从25中拿出另一个因子5, 其它的在第一次除以5后就不再包含因子5了。

代码如下:

#include<iostream>
using namespace std;

int NumZeros(int n)
{
    if(n==0)
        return 0;
    int count=0;
    while((n=n/5)>0)
    {
        count+=n;
    }
    return count;
}

int main()
{
    cout<<NumZeros(25)<<endl;
}

 

时间: 2024-10-06 10:31:08

careercup-中等难度 17.3的相关文章

OC语言实现中等难度通讯录

实现中等难度通讯录.需求: 1.定义联系⼈人类Contact.实例变量:姓名(拼⾳音,⾸首字⺟母⼤大写).性别.电话号码. 住址.分组名称.年龄.⽅方法:⾃自定义初始化⽅方法(姓名.电话号码).显⽰示联系⼈人信息 2.在main.m中定义字典,分组管理所有联系⼈人.分组名为26个⼤大写的英⽂文字⺟母. 3.可以添加联系⼈人对象,如果姓名或电话号码为空,添加失败.添加联系⼈人到匹配的分 组. 4.获取某个分组名称下所有联系⼈人,并且按照姓名升序排列. 5.从通讯录中根据电话号码搜索联系⼈人. 6.

careercup-中等难度 17.11

17.11 给定rand5(),实现一个方法rand7().也即,给定一个产生0到4(含)随机数的方法,编写一个产生0到6(含)随机数的方法. 解法: 这个函数要正确实现,则返回0到6之间的值,每个值的概率必须为1/7. 1 第一个尝试(调用次数固定) 第一个尝试时,我们可能会想产生出0到9之间的值,然后再除以7取余数.代码大致如下: int rand7() { int v=rand5()+rand5(); return v%7; } 可惜的是,上面的代码无法以相同的概率产生所有值. 方法二:

careercup-中等难度 17.6

17.6 给定一个整数数组,编写一个函数,找出索引m和n,只要将m和n之间的元素排好序,整个数组就是有序的.注意:n越小越好,也就是说,找出符合条件的最短序列. 解法: 开始解题之前,让我们先确认一下答案会是什么样的.如果要找的是两个索引,这表明数组中间有一段有待排序,其中数组开头和末尾部分是排好序的. 现在,我们借用下面的例子来解决此题: 1,2,4,7,10,11,7,12,6,7,16,18,19 首先映入脑海的想法可能是,直接找出位于开头的最长递增子序列,以及位于末尾的最长递增子序列.

careercup-中等难度 17.9

17.9 设计一个方法,找出任意指定单词在一本书中的出现频率. 解法: 1 单次查询 遍历这本书的每个单词,计算给定单词出现的次数.时间复杂度O(n),我们无法继续优化它,因为书中的每个单次都需要访问一次.当然,如果我们假设书中的单词是均匀分布的,那我们就可以只统计前半本书某个单次出现的次数,然后乘以2:或是只统计前四分之一本书某个单次出现的次数,然后乘以4. 2 多次查询 如果我们要重复执行查询,那么,或许值得我们多花些时间,多花些内存,对全书进行预处理.我们可以构造一个散列表,将单词映射到该

careercup-中等难度 17.8

17.8 给定一个整数数组(有正数和负数),找出总和最大的连续序列,并返回总和. 解法: 就是求连续子序列的和最大,不过存在一个问题: 假设整个数组都是负数,怎么样才是正确的行为呢?看看这个简单的数组{-3,-10,-5},一下答案每个都可以说的通: -3(假设子序列不能为空) 0(子序列的长度为空) INT_MIN(视为错误的情况) 我们会选择第二种(maxSum=0),但并没有所谓的"正确"答案.这一点可以跟面试官好好讨论一番.   C++实现代码: #include<ios

careercup-中等难度 17.7

17.7 给定一个整数,打印该整数的英文描述(例如"One Thousand,Two Hundred Thirty Four"). 解法: 举个例子,在转换19 323 984时,我们可以考虑分段处理,没三位转换一次,并在适当的地方插入"thousand"(千)和"million"(百万).也即, convert(19 323 984)=convert(19)+ " million "+convert(323)+"

careercup-中等难度 17.5

17.5 写一个函数来模拟游戏. 游戏规则如下: 4个槽,里面放4个球,球的颜色有4种,红(R ),黄(Y),绿(G),蓝(B).比如, 给出一个排列RGGB,表示第一个槽放红色球,第二和第三个槽放绿色球,第四个槽放蓝色球. 你要去猜这个排列.比如你可能猜排列是:YRGB.当你猜的颜色是正确的,位置也是正确的, 你就得到一个hit,比如上面第3和第4个槽猜的和真实排列一样(都是GB),所以得到2个hit. 如果你猜的颜色在真实排列中是存在的,但位置没猜对,你就得到一个pseudo-hit.比如,

careercup-中等难度 17.4

17.4 编写一个方法,找出两个数字中最大的那一个.不得使用if-else或其他比较运算符. 解法: 我们可以通过一步步的分析来将需要用到的if-else和比较操作符去掉: If a > b, return a; else, return b. If (a - b) < 0, return b; else, return a. If (a - b) < 0, 令k = 1; else, 令k = 0. return a - k * (a - b). 令z = a - b. 令k是z的最高

三星Galaxy S5全拆解 维修难度适中(组图)

中介交易 http://www.aliyun.com/zixun/aggregation/6858.html">SEO诊断 淘宝客 云主机 技术大厅 三星Galaxy S5全拆解 维修难度适中(组图) 三星Galaxy S5全拆解 维修难度适中(组图) 三星Galaxy S5全拆解 维修难度适中(组图) 三星Galaxy S5全拆解 维修难度适中(组图) 三星Galaxy S5全拆解 维修难度适中(组图) [TechWeb报道]4月11日消息,据国外媒体报道,iFixit日前对三星旗舰新机