Algorithm/BOJ

[BOJ/Python] 2559_수열

O773H 2023. 1. 5. 23:20
728x90

문제 출처 : https://www.acmicpc.net/problem/2559

 

2559번: 수열

첫째 줄에는 두 개의 정수 N과 K가 한 개의 공백을 사이에 두고 순서대로 주어진다. 첫 번째 정수 N은 온도를 측정한 전체 날짜의 수이다. N은 2 이상 100,000 이하이다. 두 번째 정수 K는 합을 구하기

www.acmicpc.net

 

문제

매일 아침 9시에 학교에서 측정한 온도가 어떤 정수의 수열로 주어졌을 때, 연속적인 며칠 동안의 온도의 합이 가장 큰 값을 알아보고자 한다.

예를 들어, 아래와 같이 10일 간의 온도가 주어졌을 때, 

3 -2 -4 -9 0 3 7 13 8 -3

모든 연속적인 이틀간의 온도의 합은 아래와 같다.

이때, 온도의 합이 가장 큰 값은 21이다. 

또 다른 예로 위와 같은 온도가 주어졌을 때, 모든 연속적인 5일 간의 온도의 합은 아래와 같으며, 

이때, 온도의 합이 가장 큰 값은 31이다.

매일 측정한 온도가 정수의 수열로 주어졌을 때, 연속적인 며칠 동안의 온도의 합이 가장 큰 값을 계산하는 프로그램을 작성하시오. 

입력

첫째 줄에는 두 개의 정수 N과 K가 한 개의 공백을 사이에 두고 순서대로 주어진다. 첫 번째 정수 N은 온도를 측정한 전체 날짜의 수이다. N은 2 이상 100,000 이하이다. 두 번째 정수 K는 합을 구하기 위한 연속적인 날짜의 수이다. K는 1과 N 사이의 정수이다. 둘째 줄에는 매일 측정한 온도를 나타내는 N개의 정수가 빈칸을 사이에 두고 주어진다. 이 수들은 모두 -100 이상 100 이하이다. 

출력

첫째 줄에는 입력되는 온도의 수열에서 연속적인 K일의 온도의 합이 최대가 되는 값을 출력한다.

 

예제 입력 1 

10 2
3 -2 -4 -9 0 3 7 13 8 -3

예제 출력 1 

21

 

예제 입력 2 

10 5
3 -2 -4 -9 0 3 7 13 8 -3

예제 출력 2 

31

 


풀이

 

연속되는 K개의 수의 부분합 중 최대값을 출력하는 문제이다.

처음 문제를 풀 때, 연속된 K개의 수의 부분합을 N-K번 새로 생성하는 코드를 작성하였으나 시간 초과가 나왔다.

 

1
2
3
4
5
6
7
8
9
10
11
12
13
import sys
 
n,k = map(int,sys.stdin.readline().split())
n_list = list(map(int,sys.stdin.readline().split()))
sum_list = []
 
for i in range(n-k):
    sum = 0
    for j in range(k):
        sum += n_list[i+j]
    sum_list.append(sum)
 
print(max(sum_list))

시간초과가 나온 코드

  1. 부분합의 리스트 sum_list를 만들고, 연속된 K개의 부분합을 N-K번 수행했다.
  2. 만들어진 부분합 중 최대값을 출력한다.
  3. 시간초과

시간을 단축하기 위해 부분합을 생성할 때, 기존 부분합의 첫 index의 값과 새로 만들 부분합의 마지막 index의 값을 바꾸는 방법으로 생성하였다. (첫 index의 값을 빼주고, 마지막 index의 값을 더해준다.)

정수들을 3 2 -2 5 4 라 하고, K=3 인 상황을 살펴보면 우선 처음 연속된 3개의 값을 더해준다.

3 2 -2 5 4

따라서 처음 생성되는 부분합은 3 + 2 + (-2) = 3 이다.

3 2 -2 5 4

다음으로 새로운 부분합을 생성할 때, 기존 부분합의 값 중, 첫번째 값인 3과 새로 만들 부분합의 마지막 값인 5을 바꾸어준다. 따라서 새로 생성되는 부분합의 값은 3 - 3 + 5 = 5 이다.

3 2 -2 5 4

마찬가지로 기존 부분합의 첫번째 값인 2를 빼주고 새로 만들 부분합의 마지막 값인 4를 더해준다.

따라서 부분합의 값은 5 - 2 + 4 = 7 이 된다.

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
import sys
 
n,k = map(int,sys.stdin.readline().split())
n_list = list(map(int,sys.stdin.readline().split()))
sum = 0
sum_list = []
 
for i in range(k):
    sum += n_list[i]
 
sum_list.append(sum)
 
for j in range(k,n):
    sum = sum - n_list[j-k] + n_list[j]
    sum_list.append(sum)
 
print(max(sum_list))
 

 

새롭게 작성된 코드는 이와 같다.

728x90