본문 바로가기

프로그래밍/알고리즘

[EP 047] 소인수분해의 소수의 갯수가 4개

반응형

문제는 이렇다.

644는 소인수분해하였을 때, 22 x 7 x 23 으로 세 개의 서로다른 소수로 이루어진다. n, n+1, n+2, n+3 이 모두 네 개의 서로다른 소수로 이루어지는 최소의 n을 구하라.

1부터 차례대로 소인수분해를 해 나가야 하기 때문에, 완전한 소수의 리스트를 관리할 수 있다. 소수판정이 매우 효율적이라서 기쁘다. 또한 소인수분해를 기록하는 리스트도 기록해 나가는데, 일단 하나의 소수만 발견하면 기록된 리스트를 이용해서 매우 효율적으로 원하는 값을 구할 수 있다. 소스를 봐라.

 

# Author : DwYoon
# PROJECT EULER
# http://projecteuler.net/index.php?section=problems&id=47
# PROBLEM 47
import time


def GetNFactors0(n, primes, _, factors):
    """
    stupid
    """
    sqrtn = int(n ** 0.5) + 1

    for p in primes:
        # p = primes[i]
        if p > sqrtn:
            break
        if n % p == 0:
            f = factors[n // p][:]
            f.append(p)
            factors.append(f)
            return len(set(f))
    # if len(factors[n]) == 1:
    primes.append(n)
    factors.append([n])
    return 1


def GetNFactors(n, primes, n_pfactors, _):
    """
    using acculumated n_pfactors array
    additionally doing prime-check
    """
    sqrtn = int(n ** 0.5) + 1

    for p in primes:
        if p > sqrtn:
            break
        if n % p == 0:
            n //= p
            if n % p == 0:
                return n_pfactors[n]
            else:
                return n_pfactors[n] + 1

    # n is primes
    primes.append(n)
    return 1


def solve1():
    solve12(GetNFactors)


def solve2():
    solve12(GetNFactors0)


def solve12(getfactor_func):
    primes = [2, 3]
    # number of prime factors
    #         for     1  2  3
    n_pfactors = [0, 0, 1, 1]
    factors = [[], [], [2], [3]]

    n = 4
    while 1:
        n_pfactors.append(getfactor_func(n, primes, n_pfactors, factors))
        #'''
        if n_pfactors[n] == n_pfactors[n - 1] == n_pfactors[n - 2] == n_pfactors[n - 3] == 4:
            print(n - 3, n - 2, n - 1, n)
            [print(e) for e in factors[-4:]]
            break
        n += 1


def main():
    t = time.time()
    solve1()
    print(f"took {time.time()-t} sec")

    t = time.time()
    solve2()
    print(f"took {time.time()-t} sec")


if __name__ == "__main__":
    main()
728x90