파이썬_실전 프로젝트

숫자 맞추기 게임 2 (스트라이크 볼 게임)

이번에는 숫자 야구라는 간단한 게임을 만들어 보겠습니다.

앞에서 만든 숫자 맞추기 게임을 조금만 수정하면 됩니다.

게임의 규칙은 사용자가 값을 입력해서, 정답과 자리가 정확하게 같으면 스트라이크, 자리는 다르지만 숫자가 존재하기만 하면 볼이라고 알려주는 규칙입니다.

정답이 195일때

123 을 입력하면, 1 스트라이크

193 을 입력하면, 2 스트라이크

951 은 3볼,

159 는 1스트라이크 2볼입니다.

 

 

1. 앞서 만든 코드 가져오기

기본적인 구조는 이전 코드와 비슷합니다.

import random
number = random.randint(1,999)
print(number)

while True:
    try:
        guess = int(input('숫자를 입력하세요 :'))
        if guess == number:
            print('정답입니다')
            break
        elif guess > number:
            print('더 작은 수 입니다')
        else:
            print('더 큰 수 입니다.')
    except:
        print('1-999중의 숫자를 입력하세요')

 

2. 숫자를 문자로 바꾸고, 자릿수는 0으로 채우기.

각 자릿수를 비교해야 하므로, 문자가 비교하기 편합니다. 발생된 난수는 문자로 바꿔주고, 사용자 입력은 원래 문자타입이므로 숫자로 변환하지 않고 그대로 둡니다.

import random
number = str(random.randint(1,999)).zfill(3) # 문자로 바꿔주고, 0으로 채움.
print(number)

 zfill() 함수는 사용자가 3자리를 입력하지 않았을 경우를 대비해서, 문자열 왼쪽의 빈칸을 0으로 채우라는 명령입니다.

 

나머지 부분은, 정답인지 확인하는 부분만 남기고 삭제해주겠습니다.

while True:
    guess = input('숫자를 입력하세요 :')  #기본타입이 문자이기 때문에, 별도로 바꿀필요가 없습니다.
    if guess == number:
        print('정답입니다')
        break
    else:
        print('틀렸습니다.')
 
3. 스트라이크, 볼 판단

스트라이크 볼을 판단하는 원리는, 자리가 완전히 일치할경우 strike 변수에 1증가, 자릿수가 달라도 숫자가 존재하면 ball 변수에 1증가 시키도록 했습니다.

import random
number = str(random.randint(1,999)).zfill(3)
print(number)

while True:
    guess = input('숫자를 입력하세요 :')
    if guess == number:
        print('정답입니다')
        break
    else:
        strike = 0
        ball = 0
        for i in range(3):
            if guess[i] == number[i]:
                strike += 1
            elif guess[i] in number:
                ball+=1
        print("스트라이크:{} 볼:{}".format(strike,ball))
195
숫자를 입력하세요 :190
스트라이크:2 볼:0
숫자를 입력하세요 :295
스트라이크:2 볼:0
숫자를 입력하세요 :951
스트라이크:0 볼:3
숫자를 입력하세요 :159
스트라이크:1 볼:2
숫자를 입력하세요 :195
정답입니다

 

4. 중복처리

위 코드에 한가지 문제가 있는데, 중복된 숫자가 있을경우, 정확하지 않은 결과가 나옵니다.

예를들어, 정답이 122일때 113을 입력하면 원래는 1스트라이크만 나와야 하지만, 현재 코드는 1스트라이크, 1볼이 나옵니다.(1이 두번 검사됩니다) 그래서 한번 스트라이크나 볼로 판단된 자리는 별도로 표시를 해줘서, 재검사가 되지 않도록 해야합니다.

이를 위해서, 임시 변수 answer(리스트타입) 를 만들어서 정답 122를 복사한 다음에( ['1','2','2'] ), answer 와 guess 를 비교해줍니다. 스트라이크나 볼판단시에, 's'와 'b'등의 숫자가 아닌 임의의 문자로 변경을 해주면, 다음번 if문에서 그자리는 무조건 false로 판단되기 때문에, 중복을 피할수 있습니다.

import random
number = str(random.randint(1,999)).zfill(3)
print(number)

while True:
    guess = input('숫자를 입력하세요 :')
    if guess == number:
        print('정답입니다')
        break

    strike = 0
    ball = 0
    answer = list(number)  # 임시 변수에 값 저장. 
    for i in range(3):
        if guess[i] == answer[i]:  # number 대신 answer과 비교
            strike += 1
            answer[i] = 's'  # 스트라이크일경우, 해당숫자를 's'로 바꿔줌.

    for i in range(3):
        if guess[i] in answer:
            ball += 1
            answer[answer.index(guess[i])] = 'b' # 위치를 모르기 때문에, 인덱스를 찾아서, 값을 바꾸어 줍니다.
    print(strike,ball, answer)

416
숫자를 입력하세요 :444
1 0 ['s', '1', '6']
숫자를 입력하세요 :145
0 2 ['b', 'b', '6']
숫자를 입력하세요 :146
1 2 ['b', 'b', 's']
숫자를 입력하세요 :416
정답입니다

 

5. 잘못된 입력 처리

입력된 숫자의 길이가 안맞거나, 숫자가 아닌 다른 문자를 입력하면 현재는 프로그램이 비정상적으로 멈추는데, 이때 사용자에게 적당한 메세지를 보여주고, 루프문을 계속 돌수 있도록 해주겠습니다.

while True:
    try:    # 에러 검사
        if .... :
            raise   # 강제로 에러를 발생시켜서, except 문으로 넘어갑니다. 
    except:
        print('에러메세지')
    else:
        # 에러 없을때 실행할 코드

기본적인 구조는 위와 같습니다. try문에서 에러를 검사하고, 에러가 있으면 except를, 에러가 없으면 else문을 실행하게 됩니다. 앞선 토픽에서의 예외처리와 다른점은, else 구문과 raise 구문입니다.

자동으로 생기지 않는 에러의 경우, 위처럼 if문으로 특정 조건을 검사한 다음 raise 문으로 강제로 에러를 발생시켜서 except의 코드를 실행하게 됩니다. 에러가 없을경우엔 나머지 else 문의 코드들을 실행하게 됩니다. 

import random
number = str(random.randint(1,999)).zfill(3)
print(number)

while True:
    guess = input('숫자를 입력하세요 :')
    
    try:
        if len(guess) != 3:   # 입력이 3자가 아니면
            raise     # 에러를 발생시킵니다.

    except :
        print('0~999 사이의 숫자를 입력해주세요')  # 에러메세지

    else:                # 에러가 없을때의 정상적인 코드
        if guess == number:
            print('정답입니다')
            break

        strike = 0
        ball = 0
        answer = list(number)
        for i in range(3):
            if guess[i] == answer[i]:
                strike += 1
                answer[i] = 's'

        for i in range(3):
            if guess[i] in answer:
                ball += 1
                answer[answer.index(guess[i])] = 'b'
        print(strike,ball)

 

숫자가 아닌 입력을 걸러낼려면, try 문에 조건문을 더 추가해주면 됩니다.

    try:
        if len(guess) != 3:
            raise

        for i in guess:             # 모든 입력이 숫자인지 검사
            if i not in '0123456789':  
                raise 

 

댓글

댓글 본문
작성자
비밀번호
  1. 쿨피
    네 맞습니다.
    초보라 어렵게 짠듯요..-.-;;
    대화보기
    • nomadlife
      아 코드 감사합니다. 살짝 어렵네요 ^^; 초보가 아니신거 같은데요? 100, -50은 스트라이크/볼을 양수음수로 구분하실려고 넣으신건가요?
      대화보기
      • 쿨피
        허접하지만 아래 처럼 해보니 중복처리 되네요 아직 초짜라서 ^^
        for j in [0,1,2]:
        TempRST = 0
        for k in [0, 1, 2]:

        if j == k and random_num[j] == guess[k]:
        Result[k][j] = 100
        elif j != k and random_num[j] == guess[k]:
        Result[k][j] = -50
        else:
        Result[k][j] = 0
        TempRST = TempRST+Result[k][j]

        print(TempRST)
        if TempRST > 0:
        STK = STK + 1
        elif TempRST < 0:
        BALL = BALL + 1
        print(Result)
        print("%d strikes and %d Balls " % (STK, BALL))
        대화보기
        • nomadlife
          제가 중복처리를 안해줬네요 뭔가 깔끔한 코드가 생각이 안서서, 그냥 저대로 뒀던듯 합니다. ^^; 혹시 아이디어 있으시면 공유 부탁드릴게요. 저도 고민해보겠습니다.
          대화보기
          • 쿨피
            number 198 guess 181 인 경우 1S1B 이어야 맞을 듯 한데 위 로직으로는 1S2B 이 나오네요,
          버전 관리
          nomadlife
          현재 버전
          선택 버전
          graphittie 자세히 보기