给定一个整数 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题 ,返回 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题_02

提示 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_hot100_03

示例 1:

输入:n = 3
输出:0
解释:3! = 6 ,不含尾随 0

示例 2:

输入:n = 5
输出:1
解释:5! = 120 ,有一个尾随 0

示例 3:

输入:n = 0
输出:0

提示:

LeetCode172. 阶乘后的零(2024秋季每日一题 1)_leetcode_04

进阶: 你可以设计并实现对数时间复杂度的算法来解决此问题吗?


暴力枚举:

可以发现在乘的过程中,末尾 0 的个数总是增加的,所以可以考虑每乘一次,就将对应的末尾 0 计数,然后消除掉末尾 0,这样可以防止溢出
比如:120 * 5 = (12 * 5)0 = 600

优化:看代码,利用 10 的倍数取余,防止溢出,并且使得末尾 0 的个数不变
比如:LeetCode172. 阶乘后的零(2024秋季每日一题 1)_算法_05,但是后面的乘积是否有0,和末尾的最后几位数有关,可以只取最后几位

时间复杂度:LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题_06
空间复杂度:LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题_07

class Solution {
public:
    int trailingZeroes(int n) {
        unsigned long long x = 1;
        long long res = 0;
        for(int i = 1; i <= n; i++){
            x *= i;
            if(x > 1000000) x = x % 1000000;
            while(x % 10 == 0) res++, x /= 10;
        }
        
        return res;
    }
};

利用数学知识推导:

LeetCode172. 阶乘后的零(2024秋季每日一题 1)_数据结构_08 的乘积,能产生末尾0的只有 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_leetcode_09 可以,所以只需要找到乘积里面有多少对LeetCode172. 阶乘后的零(2024秋季每日一题 1)_leetcode_10,并且 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题_11 的个数肯定是 多于 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_算法_12,所以只需要找乘积里有多少个 5 即可
LeetCode172. 阶乘后的零(2024秋季每日一题 1)_算法_13,像 LeetCode172. 阶乘后的零(2024秋季每日一题 1)_leetcode_14

时间复杂度:LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题_06
空间复杂度:LeetCode172. 阶乘后的零(2024秋季每日一题 1)_每日一题_07

class Solution {
public:
    int trailingZeroes(int n) {
        long long res = 0;
        for(int i = 5; i <= n; i+=5){
            int x = i;
            while(x % 5 == 0) res++, x /= 5;
        }
        
        return res;
    }
};