오일러 프로젝트 16

215 = 32768 의 각 자리수를 더하면 3 + 2 + 7 + 6 + 8 = 26 입니다. 21000의 각 자리수를 모두 더하면 얼마입니까?

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

접근

이 문제는 13번처럼 큰 정수를 다룰 수 있는지를 확인하는 문제이다. 사실 같은 수끼리 더하면 두 배가 되므로, “1” 부터 시작해서 s_add() 함수에 같은 값을 두 번 넣어서 두 배하는 동작을 1,000번 하면 간단히 해결된다.

파이썬은 큰 정수를 지원하므로 2의 1000 제곱을 계산해서 문자열로 바꾼 후, 다시 각 숫자를 정수로 바꿔서 합산한다.

sum(map(int, str(2**1000)))
# 1366

어떤 수를 두 배 하는 것은 같은 수 두 개를 더하는 것과 같으므로, 문자열로 큰 수를 더할 수 있는 s_add() 함수를 사용해서 아래와 같이 풀어도 된다.

_parse = lambda s: [int(c) for c in s[::-1]]
_comps = lambda xs: ''.join(chr(x + 48) for x in xs[::-1])

def s_add(*args: str) -> str:
  l = max(map(len, args))
  xs = list(map(_parse, (x.zfill(l) for x in args)))
  temp, z = [], 0
  for ws in zip(*xs):
    z, w = divmod(sum(ws) + z, 10)
    temp.append(w)
  if z:
    temp.append(z)
  return _comps(temp)


a = "1"
for _ in range(1000):
  a = s_add(a, a)

print(sum(map(int, a)))

문자열을 이용한 큰 정수의 곱셈

문자열을 이용해서 큰 정수의 덧셈을 구현한 s_add()와 비슷하게 s_multi() 함수를 작성하고, 이를 이용해서 거듭제곱을 계산하는 s_power() 함수를 구현해보자.

곱셈을 구현하는 것도 그리 어렵지 않다. A x B 를 계산할 때, B의 각 자리 숫자를 b1, b2, b3, … bn 이라 하면 A * bn, A * b1 * 10, A * b2 * 100… 을 각각 문자열로 계산한 다음, s_add() 함수를 이용해서 이들을 모두 더하면 결과를 얻을 수 있다.


def s_multi(a: str, b: str) -> str:
  res: list[str] = []
  a, b = map(_parse, (a, b))
  for lv, y in enumerate(b):
    temp = [0] * lv  # 자리수에 맞게 0을 추가해줌
    z = 0
    for x in a:
      z, w = divmod(x * y + z, 10)
      temp.append(w)
    if z > 0:
      temp.append(z)
    res.append(_comps(temp))
  return s_add(*res)


# 제곱을 이용한 pow 함수

def s_pow(a: str, b: str|int) -> str:
  if isintance(b, str):
    b = int(b)
  if b == 0:
    return "1"
  if b == 1:
    return a
  # a의 (b /2) 제곱을 계산하고 이를 제곱한다.
  # 이렇게하여 반복주기를 줄인다.
  r = s_pow(a, b // 2)
  if b % 2 == 0:
    return s_multi(r, r)
  return s_multi(s_multi(r, r), a)

print(sum)map(int, s_power("2", 1000))))
# 1366

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