개인 실습: 08-04

정지호·2022년 8월 4일
0

개인 실습 진행

목록 보기
11/41

파이썬 실습(프로그래머스)

다트

-문제: 카카오톡 게임별의 하반기 신규 서비스로 다트 게임을 출시하기로 했다. 다트 게임은 다트판에 다트를 세 차례 던져 그 점수의 합계로 실력을 겨루는 게임으로, 모두가 간단히 즐길 수 있다.

갓 입사한 무지는 코딩 실력을 인정받아 게임의 핵심 부분인 점수 계산 로직을 맡게 되었다. 다트 게임의 점수 계산 로직은 아래와 같다.

1. 다트 게임은 총 3번의 기회로 구성된다.
2. 각 기회마다 얻을 수 있는 점수는 0점에서 10점까지이다.
3. 점수와 함께 Single(S), Double(D), Triple(T) 영역이 존재하고 각 영역 당첨 시 점수에서 1제곱, 2제곱, 3제곱 (점수1 , 점수2 , 점수3 )으로 계산된다.
4. 옵션으로 스타상() , 아차상(#)이 존재하며 스타상() 당첨 시 해당 점수와 바로 전에 얻은 점수를 각 2배로 만든다. 아차상(#) 당첨 시 해당 점수는 마이너스된다.
5. 스타상()은 첫 번째 기회에서도 나올 수 있다. 이 경우 첫 번째 스타상()의 점수만 2배가 된다. (예제 4번 참고)
6. 스타상()의 효과는 다른 스타상()의 효과와 중첩될 수 있다. 이 경우 중첩된 스타상() 점수는 4배가 된다. (예제 4번 참고)
7. 스타상(
)의 효과는 아차상(#)의 효과와 중첩될 수 있다. 이 경우 중첩된 아차상(#)의 점수는 -2배가 된다. (예제 5번 참고)
8. Single(S), Double(D), Triple(T)은 점수마다 하나씩 존재한다.
9. 스타상(), 아차상(#)은 점수마다 둘 중 하나만 존재할 수 있으며, 존재하지 않을 수도 있다.
0~10의 정수와 문자 S, D, T,
, #로 구성된 문자열이 입력될 시 총점수를 반환하는 함수를 작성하라.


나의 풀이(결국 못풀었다. 몇 시간을 썼는데도......)

def solution(dartResult):

    answer = 0
    answer_list = []
    if "10" not in dartResult: # 10이 없는 경우
        li = list(dartResult) # 바로 리스트화
    else: # 10 이 있는 경우
        li = list(dartResult)
        li.insert(dartResult.index("10"),"10") #10을 집어넣고
        del li[li.index("10")+1] # 1제거
        del li[li.index("10")+1] # 0제거

    li_1 = []
    for i in li:
        try:
            li_1.append(int(i)) # 정수화시켜 리스트에 넣음
        except ValueError: #정수화가 되지 않는다면
            li_1.append(i) # 문자 그대로 넣음
    for i, element in enumerate(li_1):
        if type(element) == int:
            count += 1
            if (li_1[i+1] == "S" and i+1 <= len(li_1)):
                if li_1[i+2] == "*" and i+2 <= len(li_1):
                    if i==0:
                        answer_list.append(element*2)
                    elif i==3:
                        answer_list[0] *= 2
                        answer_list.append(element*2)
                    elif i==6:
                        answer_list[1] *= 2
                        answer_list.append(element*2)
                elif li_1[i+2] == "#" and i+2 <= len(li_1):
                        answer_list.append(-element)
                else:
                    answer_list.append(element)
                    
                
            elif (li_1[i+1] == "D" and i+1 <= len(li_1)):
                if li_1[i+2] == "*" and i+2 <= len(li_1):
                    if i==0:
                        answer_list.append((element**2)*2)
                    elif i==3:
                        answer_list[0] *= 2
                        answer_list.append((element**2)*2)
                    elif i==6:
                        answer_list[1] *= 2
                        answer_list.append((element**2)*2)
                elif li_1[i+2] == "#" and i+2 <= len(li_1):
                        answer_list.append(-element**2)
                else:
                    answer_list.append(element**2)
            elif (li_1[i+1] == "T" and i+1 <= len(li_1)):
                if li_1[i+2] == "*" and i+2 <= len(li_1):
                    if i==0:
                        answer_list.append((element**3)*2)
                    elif i==3:
                        answer_list[0] *= 2
                        answer_list.append((element**3)*2)
                    elif i==6:
                        answer_list[1] *= 2
                        answer_list.append((element**3)*2)
                elif li_1[i+2] == "#" and i+2 <= len(li_1):
                        answer_list.append(-element**3)
                else:
                    answer_list.append(element**3)
# 각 조건에 따라 answer_list 에 맞는 값을 넣었다.            
    for i in range(len(answer_list)):
        answer += answer_list[i] # 뻘짓이다. 그냥 sum(answer_list) 쓰면 된다.
    return answer
  • 풀이 방법 자체는 틀린 것은 아니었다.
  • 하지만 인덱스 범위 초과 문제가 계속 발생하였고, 이를 해결하기만 하면 되었지만 이를 해결하는 것이 너무 어려웠다. 결국 다른 사람의 코드를 참고하였다.

정답(다른사람의 코드)

def solution(dartResult):
    n = '' # 빈 문자열. 숫자를 여기에 저장한다.
    score = [] # 세 점수를 이 리스트에 넣는다.
    for i in dartResult:
        if i.isnumeric(): # 주어진 문자열이 숫자로 되어있는지 검사하는 함수. 숫자로 판별되면
            n += i # 빈 문자열에 숫자모양 문자를 저장 
        elif i == 'S': # Single이면 
            n = int(n)**1 # 저장된 문자를 정수화하고 1제곱한 후,
            score.append(n) # 스코어 리스트에 넣어준다. 
            n = '' # 초기화
        elif i == 'D': # Double 이면
            n = int(n)**2 # 저장된 문자를 정수화하고 2제곱한 후,
            score.append(n) # 스코어 리스트에 넣어준다. 
            n = '' # 초기화
        elif i == 'T': # Triple 이면
            n = int(n)**3 #저장된 문자를 정수화하고 3제곱한 후,
            score.append(n) # 스코어 리스트에 넣어준다.
            n = '' # 초기화
        elif i == '*': # 스타상이면
            if len(score) > 1: # 첫 점수가 아니라면
                score[-2] = score[-2] * 2 # 전에 점수 두 배
                score[-1] = score[-1] * 2 # 해당 점수 두 배
            else:  # 첫 점수라면
                score[-1] = score[-1] * 2 # 해당 점수만 두 배
        elif i == '#': # 아차상이면
            score[-1] = score[-1] * -1 # 해당 점수 음수
        
    return sum(score) #리스트 요소들 총 합치기
  • isnumeric() 함수: 주어진 문자열이 숫자로 되어있는지 검사하는 함수이다. 숫자값 표현에 해당하는 문자열까지 인정하는 함수로써, 제곱근 및 분수, 거듭제곱 특수문자도 숫자로 판별.
  • sum(리스트명) -> 리스트 요소들 다 더해주기 (for 문 쓸 필요가 없다.)
  • 여기서 중요 아이디어가 '초기화' 인 듯 싶다. S,D,T에 따라 score 리스트에 값을 저장해준 후, n을 초기화 해주면 단순하게 풀어나갈 수 있다. 스타상, 아차상은 score 리스트 값만 건드리면 된다.

++ isdigit() 함수: 주어진 문자열이 숫자로 되어있는지 검사하는 함수. 단일 글자가 숫자 모양으로 생겼으면 무조건 숫자로 판별
++ isdecimal() 함수: 주어진 문자열이 int형으로 변환이 가능한지 알아내는 함수. 특수문자 중 숫자모양을 숫자로 치지 않는다. 예를 들어 거듭 제곱은 특수문자 이면서 숫자모양이므로, 이 함수에서 숫자로 판별되지 않는다.


참고: https://it-neicebee.tistory.com/33 /
https://velog.io/@godiva7319/%ED%94%84%EB%A1%9C%EA%B7%B8%EB%9E%98%EB%A8%B8%EC%8A%A4-Level1-1%EC%B0%A8-%EB%8B%A4%ED%8A%B8-%EA%B2%8C%EC%9E%84-Python

profile
정지호

0개의 댓글