N-ta cyfra — Leetcode Problem 400

Dec 10 2022
Oryginalny link do problemu: https://leetcode.com/problems/nth-digit/description/ W tym zadaniu jesteśmy proszeni o znalezienie n-tej cyfry nieskończonej sekwencji zapisanej jako ciąg 1,2,3,4,5 ,6,7,8,9… Na przykład trzecią cyfrą będzie 3, ale dziesiątą cyfrą będzie 1 (miejsce dziesiątek z 10), a 11 będzie 0 (miejsce jedności z 10).

Oryginalny link do problemu:https://leetcode.com/problems/nth-digit/description/

W tym zadaniu jesteśmy proszeni o znalezienie n-tej cyfry nieskończonego ciągu zapisanego jako ciąg 1,2,3,4,5,6,7,8,9…

Na przykład trzecią cyfrą będzie 3, ale dziesiątą cyfrą będzie 1 (miejsce dziesiątek z 10), a 11. to 0 (miejsce jedności z 10).

Zanim przejdziemy do wyjaśnień, szybkie podejście, które pokonuje 100% zgłoszeń, jest następujące, ale musimy użyć odrobiny matematyki:

class Solution {
    public int findNthDigit(int n) {
        if (n < 10) {
            return n;
        }
        long numOfDigits = 0;
        long power = 1;
        while (numOfDigits + (9 * (long)Math.pow(10, power - 1) * power) <= n) {
            numOfDigits += (9 * (long)Math.pow(10, power - 1) * power);
            power++;
        }
        long quotient = (n - numOfDigits) / power;
        long mod = (n - numOfDigits) % power;
        long i = mod == 0 ? (quotient + (long)Math.pow(10, power - 1) - 1) : (quotient + (long)Math.pow(10, power - 1));
        String num = String.valueOf(i);
        return mod == 0 ? Character.digit(num.charAt(num.length() - 1), 10) : Character.digit(num.charAt((int)mod - 1), 10);
    }
}

Więc przeszedłem do przeglądania i obliczania poprawnej liczby, z której pochodziłaby n-ta cyfra, tak jak poniżej:

class Solution {
    public int findNthDigit(int n) {
        int i = 0;
        int prevIndex = 0;
        int index = 0;
        while (index < n) {
            i++;
            String num = String.valueOf(i);
            prevIndex = index;
            index += num.length();
        }
        String str = String.valueOf(i);
        return Character.digit(str.charAt(n - prevIndex - 1), 10);
    }
}

Sposobem na przyspieszenie i akceptację było utworzenie rozwiązania O (log n) poprzez znalezienie wzorca liczby cyfr generowanych przez każdy zakres liczb, jak następuje

Single digits 1-9 => 9 * 10^0 * 1 digits
Double digits 10-99 => 9 * 10^1 * 2 digits
...so the general formula is 9 * 10^(m - 1) * m digits for each number range

class Solution {
    public int findNthDigit(int n) {
        if (n < 10) {
            return n;
        }
        long numOfDigits = 0;
        long power = 1;
        while (numOfDigits + (9 * (long)Math.pow(10, power - 1) * power) <= n) {
            numOfDigits += (9 * (long)Math.pow(10, power - 1) * power);
            power++;
        }
        long quotient = (n - numOfDigits) / power;
        long mod = (n - numOfDigits) % power;
        long i = mod == 0 ? (quotient + (long)Math.pow(10, power - 1) - 1) : (quotient + (long)Math.pow(10, power - 1));
        String num = String.valueOf(i);
        return mod == 0 ? Character.digit(num.charAt(num.length() - 1), 10) : Character.digit(num.charAt((int)mod - 1), 10);
    }
}