• C++二分查找算法:阶乘函数后 K 个零


    本文涉及的基础知识点

    二分查找算法合集

    题目

    f(x) 是 x! 末尾是 0 的数量。回想一下 x! = 1 * 2 * 3 * … * x,且 0! = 1 。
    例如, f(3) = 0 ,因为 3! = 6 的末尾没有 0 ;而 f(11) = 2 ,因为 11!= 39916800 末端有 2 个 0 。
    给定 k,找出返回能满足 f(x) = k 的非负整数 x 的数量。
    示例 1:
    输入:k = 0
    输出:5
    解释:0!, 1!, 2!, 3!, 和 4! 均符合 k = 0 的条件。
    示例 2:
    输入:k = 5
    输出:0
    解释:没有匹配到这样的 x!,符合 k = 5 的条件。
    示例 3:
    输入: k = 3
    输出: 5
    参数范围
    0 <= k <= 109

    分析

    时间复杂度

    O(logn*log5n)。FirstEqualMore内的循环logn次,GetNum内循环log5N。

    0个数

    0的个数就是2和5的个数的较小者。5的个数一定不会多余2的个数,所以计算5的个数就可以了。
    如果5x<=value,那么2x一定小于value。
    除非是0个,否则5的个数一定不会等于2的个数。
    令x>0,则5x/2 = 2x+x/2 >2x > x

    5个数

    初步想法

    非25的倍数,5的倍数+1
    非125的倍数,25的倍数+2
    非625的倍数,125的倍数+3

    可以这样想

    5的倍数+1
    25的倍数+1
    125的倍数+1

    二分

    初:寻找第一个和最后一个x,使得f(x)等于k,两者相减再+1。要特殊处理不存在f(x)等于k。所以改成寻找x1=第一个f(x)大于等于k,x2=第一个f(x)大于k,x2也是第一个f(x)大于等于k+1。

    代码

    核心代码

    class Solution {
    public:
    int preimageSizeFZF(int k) {
    return FirstEqualMore(k + 1) - FirstEqualMore(k);
    }
    int FirstEqualMore(int k)
    {
    long long left = -1, right = k * 5LL;
    while (right - left > 1)
    {
    const auto mid = left + (right - left) / 2;
    if (GetNum(mid) >= k)
    {
    right = mid;
    }
    else
    {
    left = mid;
    }
    }
    return right;
    }
    long long GetNum(long long llVaue)
    {
    long long llRet = 0;
    long long five = 5;
    while (five <= llVaue)
    {
    llRet += llVaue / five;
    five *= 5;
    }
    return llRet;
    }
    };

    测试用例

    template
    void Assert(const T& t1, const T& t2)
    {
    assert(t1 == t2);
    }

    template
    void Assert(const vector& v1, const vector& v2)
    {
    if (v1.size() != v2.size())
    {
    assert(false);
    return;
    }
    for (int i = 0; i < v1.size(); i++)
    {
    Assert(v1[i], v2[i]);
    }
    }

    int main()
    {
    vector grid;
    int res = 0;
    {
    Solution slu;
    res = slu.preimageSizeFZF(0);
    Assert(5, res);
    }
    {
    Solution slu;
    res = slu.preimageSizeFZF(5);
    Assert(0, res);
    }
    {
    Solution slu;
    res = slu.preimageSizeFZF(3);
    Assert(5, res);
    }

    //CConsole::Out(res);
    
    • 1

    }

    2023年3月旧代码

    如果f(x)等于k,则f(x+1)、f(x+2)、f(x+3)、f(x+4)都等于k。如果不存在f(x)等于k,则结果为0。所以只有两种返回值,5或0。
    class Solution {
    public:
    int preimageSizeFZF(int k) {
    int left = 0, right = 1000 * 1000 * 1000 + 1;
    while (right > left + 1)
    {
    int iMid = left + (right - left) / 2;
    const int iNum = GetFiveNum(iMid);
    if (iNum == k)
    {
    return 5;
    }
    else if (iNum < k)
    {
    left = iMid;
    }
    else
    {
    right = iMid;
    }
    }
    return (GetFiveNum(left) == k) ? 5 : 0;
    }
    //获取[0,iMax*5] 质因数5的个数
    int GetFiveNum(int iMax)
    {
    int iNum = iMax;
    int tmp = 5;
    while (iMax >= tmp)
    {
    iNum += iMax / tmp;
    tmp *= 5;
    }
    return iNum;
    }
    };

    扩展阅读

    视频课程

    有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
    https://edu.csdn.net/course/detail/38771

    如何你想快速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
    https://edu.csdn.net/lecturer/6176

    相关下载

    想高屋建瓴的学习算法,请下载《闻缺陷则喜算法册》doc版
    https://download.csdn.net/download/he_zhidan/88348653

    洒家想对大家说的话
    闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
    墨家名称的来源:有所得以墨记之。
    如果程序是一条龙,那算法就是他的是睛

    测试环境

    操作系统:win7 开发环境: VS2019 C++17
    或者 操作系统:win10 开发环境: VS2022 C++17

  • 相关阅读:
    facebook审核流程
    day4-聚类文章CONVEX BICLUSTERING
    10.Tomcat,Servlet
    现有n1+n2种面值的硬币,其中前n1种为普通币,可以取任意枚,后n2种为纪念币,每种最多只能取一枚,每种硬币有一个面值,问能用多少种方法拼出m的面值?
    RTMP摄像头直播-CameraX数据采集处理
    最近找到的几个不错的手机APP
    ACWing471. 棋盘-DFS剪枝
    微信小程序连接阿里云快速入门【物联网】
    Spring Boot 如何使用Liquibase 进行数据库迁移
    布隆过滤器原理很好懂
  • 原文地址:https://blog.csdn.net/he_zhidan/article/details/134295497