알고리즘/프로그래머스

[프로그래머스] 파이썬 _ 수식 최대화 (2020 KAKAO INTERN)

이숭간 2021. 7. 5. 21:34
728x90

https://programmers.co.kr/learn/courses/30/lessons/67257

 

코딩테스트 연습 - 수식 최대화

IT 벤처 회사를 운영하고 있는 라이언은 매년 사내 해커톤 대회를 개최하여 우승자에게 상금을 지급하고 있습니다. 이번 대회에서는 우승자에게 지급되는 상금을 이전 대회와는 다르게 다음과

programmers.co.kr

문제유형 : 구현, 리스트?

 

문제풀이 :

  • 나는 큐 2개를 이용해서 풀었다.

 

정답코드

# 연산자의 우선순위를 재정의하여 만들수있는 절댓값이 가장 큰 숫자 
# 주어지는 숫자는 모두 양수 
from itertools import permutations
from collections import deque


def solution(expression):
    answer= 0
    # 0. 숫자와 연산자로 분리
    number = ''
    op = set()
    q = deque([])
    for i in expression:
        if i.isdigit():
            number += i
        else:
            q.append(number)
            q.append(i)
            op.add(i)
            number = ''
    q.append(number)

    # 1. 주어진 문자열에 수식이 몇개있는지 확인
    len_op = len(op)
    op = list(op)
    # 2. 수식의 가능한 우선순위를 모두 구한다.
    for case in permutations(op, len_op):

        # 3. 해당 우선순위대로 계산했을때 나온 값들을 이전계산값과 비교하여 더큰값으로 계쏙 저장한다.
        new_q = deque([])
        _q = q.copy() # 새로운 우선순위가 나오면 새롭게 계산해야 하므로 복사해주어야함
        # 3.1. 하나의 연산자마다 전체식을 한번 돌아서 새로운 식을 만든다고 생각
        for i in range(len_op):
            # 큐가 빌때까지
            while _q:
                curr = _q.popleft()
                # 현재 우선순위에 해당하는 연산자라면 
                if curr == case[i]:
                    before = new_q.pop()
                    after = _q.popleft()
                    value = str(eval(before + case[i] + after))
                    new_q.append(value)
                else:
                    new_q.append(curr)
            # 현재 case에 대해 계산이 모두 완료된경우 최대값을 비교하여 갱신 
            if len(new_q) == 1:
                answer = max(answer, abs(int(new_q[-1])))
                break
            _q = new_q.copy()  # 대상이 되는 큐를 새로운 식으로 바꿔준다. 이떄도 꼭 copy를 해줘야함 
            new_q = deque([]) # 위에서 copy를 안하면 이게 반영이됨, 새로운 식을 저장할 공간 초기화 
    print(answer)
    return answer

solution("100-200*300-500+20")

 

기억하자 :

  • 파이썬에서 리스트를 다른 변수에 저장할때 레퍼런스가 넘어가서 예상하지 못한 결과가 나올때가 종종 생긴다... 정확히 머리로 설계한뒤에 구현하자.
  • 문자열로된 식처럼 숫자+문자가 이어져있는걸 한방에 숫자, 문자 리스트로 쪼개는 방법 - by 정규식
    • ex = re.split(r'(\D)',expression) 
  • 더 쉬운 다른분 풀이 
  •  연산자의 인덱스를 찾아서 이용!!
    • 현재 우선순위인 연산자를 만나면 해당 인덱스 앞뒤의 값을 가져온후에 연산자 인덱스-1 위치에 해당계산값을 넣어준다.
    • 그후에 연산자인덱스위치, 연산자인덱스+1 위치 값은 없애줘야하므로 리스트 슬라이싱을 이용해 두개값을 없애서 다시 리스트를 이어붙인다.
  • def solution(expression):
        #1
        op = [x for x in ['*','+','-'] if x in expression]
        op = [list(y) for y in permutations(op)]
        ex = re.split(r'(\D)',expression)
    
        #2
        a = []
        for x in op:
            _ex = ex[:]
            for y in x:
                while y in _ex:
                    tmp = _ex.index(y)
                    _ex[tmp-1] = str(eval(_ex[tmp-1]+_ex[tmp]+_ex[tmp+1]))
                    _ex = _ex[:tmp]+_ex[tmp+2:]
            a.append(_ex[-1])
    
        #3
        return max(abs(int(x)) for x in a)