오일러 프로젝트 55

47이란 숫자를 골라서 뒤집은 다음 다시 원래 수에 더하면 47 + 74 = 121과 같이 대칭수(palidrome)가 됩니다. 물론 모든 숫자가 이토록 쉽게 대칭수를 만들어내지는 않습니다. 예를 들어 349의 경우에는

  • 349 + 943 = 1292
  • 1292 + 2912 = 4213
  • 4213 + 3124 = 7337

위에서 보는 것처럼 3번의 반복과정을 거쳐야 대칭수가 됩니다. 196과 같은 몇몇 숫자들은 이와 같은 과정을 아무리 반복해도 대칭수가 되지 않을 것이라고 추측되는데, 이런 수를 라이크렐 수라고 부릅니다. 아직 증명되지는 않았지만, 문제 풀이를 위해서 일단 라이크렐 수가 존재한다고 가정을 하겠습니다.

또한 1만 이하의 숫자들은 50번 미만의 반복으로 대칭수가 되든지 라이크렐 수이든지 둘 중 하나라고 합니다. 1만을 넘어서면 10677에 이르렀을 때 비로소 53번의 반복으로 4668731596684224866951378664라는 28자리의 대칭수가 만들어집니다.

그러면 1만 이하에는 몇 개의 라이크렐 수가 존재합니까?

http://euler.synap.co.kr/prob_detail.php?id=55

접근

문제가 길지만 사실 간단한 내용이다. 1만 이하의 수 중에서 라이크렐 수의 개수를 세면 된다. 어떤 수가 라이크렐 수인지를 판별하는 함수를 하나 작성하자. 라이크렐 수는 다음과 같이 판단한다.

  1. 원래의 수와 그 숫자를 뒤집은 수를 더한다.
  2. 더한 결과가 대칭수(palindrome)인지 판단한다. 대칭수가 되면 라이크렐 수가 아니다.
  3. 1~2의 과정을 50회까지 반복하는 동안 대칭수가 아니라면 라이크렐 수로 판정한다.

이 과정을 코드로 옮기면 되기에 매우 간단하다.

def test(n: int) -> bool:
  for _ in range(50):
    n += int(str(n)[::-1])
    if str(n) == str(n)[::-1]:
      return False
  return True

print(sum(1 for n in range(10_000) if test(n + 1)))

배열을 사용하는 방법

큰 수를 사용하지 않으려면 이 문제를 어떻게 풀 수 있을까? 정수를 각 자리 숫자(정수)의 배열로 표현하면 처리할 수 있다. 덧셈을 구현하는 부분이 조금 까다롭게 느껴질 수 있지만, 더하는 두 수가 자릿수가 같고, 거꾸로 된 수이기 때문에 역순으로 더해도 결과가 동일하기 때문에 예전에 작성해본 s_add() 함수보다 오히려 간단할 수 있다.

from math import log10

def digits(n: int) -> list[int]:
    '''정수를 각 자릿수 숫자의 배열로 변환'''
    l = int(log10(n) + 1)
    return [(n // 10**(l-i-1)) % 10 for i in range(l)]

def ispal(xs):
    '''정수배열이 대칭인지를 검사'''
    l = round(len(xs) / 2)
    return xs[:l] == xs[len(xs)-1:-l-1:-1]

def test(n):
    ds = digits(n)
    for _ in range(50):
        es = []
        z = 0
        for x, y in zip(ds, ds[::-1]):
            z, w = divmod(x + y + z, 10)
            es.append(w)
        if z > 0:
            es.append(z)
        if ispal(es):
            return False
        ds = es[::-1]
    return True

print(sum(1 for n in range(10000) if test(n + 1)))

Read more

워드프레스에서 고스트로 이전

워드프레스에서 고스트로 이전

이 글을 쓰면서도 믿기 힘든 사실인데, 블로그라는 걸 처음 시작한지가 20년이 되었습니다. 이글루스에서 처음 시작했다가, SK컴즈가 인수한다고 발표함과 동시에 워드프레스로 플랫폼을 옮겼죠. 워드프레스오 옮긴 이후에는 호스팅 환경을 이리 저리 옮기긴 했지만 거의 18년 가까이 워드프레스를 사용해온 것 같습니다. 그 동안 워드프레스는 블로깅 툴에서 명실상부한 범용CMS로 발전했습니다. 사실 웬만한 홈페이지들은 이제

By sooop
띄어쓰기에 대한 생각

띄어쓰기에 대한 생각

업무 메일을 쓸 때 가장 많이 쓰는 말 중에 하나가 메일 말미에 ‘업무에 참고 부탁 드립니다.‘인데요, 어느 날부터 아웃룩에서 이 ‘부탁 드립니다’가 틀렸다고 맞춤법 지적을 하기 시작했습니다. 맞는 말은 ‘부탁드립니다’라고 붙여 쓰는 거라고. 사실 아래아한글 시절부터 이전의 MS워드까지, 워드프로세서들의 한국어 맞춤법 검사 실력은 거의 있으나 마나 한

By sooop

구글 포토에서 아이클라우드로 탈출한 후기

한 때 구글 포토가 백업 용량을 무제한으로 제공해 주겠다고해서, 구글 포토를 사용해서 사진을 백업해왔습니다. 물론 이 이야기의 결말은 저나 이 글을 읽고 있는 여러분이나 모두 알고 있습니다. 사실 AI에게 학습 시킬 이미지 데이터를 모으기 위한 것일 뿐이라거나 하는 이야기는 그 당시에도 있었습니다만, 에이 그래도 구글인데 용량은 넉넉하게 주겠지…하는 순진한

By sooop

Julia의 함수 사용팁

연산자의 함수적 표기 Julia의 연산자는 기본적으로 함수이며, 함수 호출 표기와 같은 방식으로 호출하는 것이 가능합니다. 또한 그 자체로 함수이기 때문에 filter(), map() 과 같이 함수를 인자로 받는 함수에도 연산자를 그대로 적용하는 것이 가능합니다. 특히 + 연산자는 sum() 함수와 같이 여러 인자를 받아 인자들의 합을 구할 수 있습니다. 2 + 3 # = 5 +(2,

By sooop