티스토리 뷰
- 오전
- 운동
- 오후
- 코딩테스트
- 저녁
- 컨셉아트 짜보기
코딩테스트 - 동영상 재생기
링크 : https://school.programmers.co.kr/learn/courses/30/lessons/340213
당신은 동영상 재생기를 만들고 있습니다. 당신의 동영상 재생기는 10초 전으로 이동, 10초 후로 이동, 오프닝 건너뛰기 3가지 기능을 지원합니다. 각 기능이 수행하는 작업은 다음과 같습니다.
10초 전으로 이동: 사용자가 "prev" 명령을 입력할 경우 동영상의 재생 위치를 현재 위치에서 10초 전으로 이동합니다. 현재 위치가 10초 미만인 경우 영상의 처음 위치로 이동합니다. 영상의 처음 위치는 0분 0초입니다.
10초 후로 이동: 사용자가 "next" 명령을 입력할 경우 동영상의 재생 위치를 현재 위치에서 10초 후로 이동합니다. 동영상의 남은 시간이 10초 미만일 경우 영상의 마지막 위치로 이동합니다. 영상의 마지막 위치는 동영상의 길이와 같습니다.
오프닝 건너뛰기: 현재 재생 위치가 오프닝 구간(op_start ≤ 현재 재생 위치 ≤ op_end)인 경우 자동으로 오프닝이 끝나는 위치로 이동합니다.
동영상의 길이를 나타내는 문자열 video_len, 기능이 수행되기 직전의 재생위치를 나타내는 문자열 pos, 오프닝 시작 시각을 나타내는 문자열 op_start, 오프닝이 끝나는 시각을 나타내는 문자열 op_end, 사용자의 입력을 나타내는 1차원 문자열 배열 commands가 매개변수로 주어집니다. 이때 사용자의 입력이 모두 끝난 후 동영상의 위치를 "mm:ss" 형식으로 return 하도록 solution 함수를 완성해 주세요.
문제가 너무 길다.
일단 정리하자면 동영상 재생기에는 2개의 기능 "Next"와 "Prev"가 있고
next는 10초 후로 보내고
prev는 10초 뒤로 보내는 기능이다.
기본적으로 이 비디오 재생기에는 "오프닝 건너뛰기"라는 기능을 하는데
만약 내 위치(pos)가 오프닝 구간에 있다면
op_end 로 건너뛰면 되는거다.
구현해보면 이렇다.
public string solution(string video_len, string pos, string op_start, string op_end, string[] commands)
{
int intPos = ToInt(pos);
int intOpStart = ToInt(op_start);
int intOpEnd = ToInt(op_end);
int intVideoLen = ToInt(video_len);
//오프닝에 있다면 건너뛰기
if (intPos >= intOpStart && intPos <= intOpEnd)
{
intPos = intOpEnd;
}
//커맨드에 따라 움직이기 시작
foreach (string command in commands)
{
switch (command)
{
case "next":
//10초 후가 오프닝 시작 후라면, 10초 후가 오프닝 끝나기 전 이라면
if (intPos + 10 >= intOpStart && intPos + 10 <= intOpEnd)
{
intPos = intOpEnd;
}
else
{
intPos = Math.Min(intPos + 10, intVideoLen);
}
break;
case "prev":
//10초 전이 오프닝 끝나기 전 이라면, 10초전이 오프닝 시작 후 라면
int newPos = Math.Max(0, intPos - 10);
if (newPos >= intOpStart && newPos <= intOpEnd)
{
intPos = intOpEnd;
}
else
{
intPos = Math.Max(0, intPos - 10);
}
break;
default:
Console.WriteLine($"알 수없는 명령어 {command}");
break;
}
}
return ToString(intPos);
}
public int ToInt(string time)
{
string[] parts = time.Split(':');
int minutes = int.Parse(parts[0]);
int seconds = int.Parse(parts[1]);
return (minutes * 60) + seconds;
}
public string ToString(int time)
{
int minutes = time / 60;
int seconds = time % 60;
return $"{minutes:D2}:{seconds:D2}";
}
나도 코드를 길게 써버렸다
일단 "00:00"는 string값이니 수 계산을 위해 int로 바꿔주는 함수와
return할땐 다시 string이어야하니 string으로 바꿔주는 함수를 구현했다.
//오프닝에 있다면 건너뛰기
if (intPos >= intOpStart && intPos <= intOpEnd)
{
intPos = intOpEnd;
}
오프닝을 건너뛰기위한 함수를 구현하고
//커맨드에 따라 움직이기 시작
foreach (string command in commands)
{
switch (command)
{
case "next":
//10초 후가 오프닝 시작 후라면, 10초 후가 오프닝 끝나기 전 이라면
if (intPos + 10 >= intOpStart && intPos + 10 <= intOpEnd)
{
intPos = intOpEnd;
}
else
{
intPos = Math.Min(intPos + 10, intVideoLen);
}
break;
case "prev":
//10초 전이 오프닝 끝나기 전 이라면, 10초전이 오프닝 시작 후 라면
int newPos = Math.Max(0, intPos - 10);
if (newPos >= intOpStart && newPos <= intOpEnd)
{
intPos = intOpEnd;
}
else
{
intPos = Math.Max(0, intPos - 10);
}
break;
default:
Console.WriteLine($"알 수없는 명령어 {command}");
break;
}
}
foreach문으로 받은 커맨드들을 하나씩 돌려가며
switch 문으로 커맨드에 따라 작동하도록 만들었다.
이 문제는 특히 오랜 시간이 걸린 것 같다
막혔던 원인에 대해서 하나씩 파헤쳐보자면
1. 문제 잘 안읽음.
문제 중 이런 설명이 있었다.
op_start ≤ 현재 재생 위치 ≤ op_end
이를 안읽고 op_start < 현재 재생위치 < op_end로 판단하고 있어서
오프닝에 딱 걸쳐져 있다면 작동하지 건너뛰기가 작동하지않는 오류가 생겼었다.
2. 동영상에서 -00:05은 없음.
case "prev":
//오프닝 안에 있다면
if (intPos - 10 >= intOpStart && intPos - 10 <= intOpEnd)
{
intPos = intOpEnd;
}
else
{
intPos = Math.Max(0, intPos - 10);
}
break;
오프닝 구간이 00:00~00:05인 반례가있었다.
pos는 00:03 그렇다면 오프닝 구간 안에 있는 것이다.
첫번째 오프닝 건너뛰기를 한pos는 00:05
prev가 작동하면 00:00이 되어야하지만
위 코드는 pos를 마이너스 값으로 만든다 (ex : -00:05)
00:00은 오프닝 안에 포함된 값이 맞지만
-00:05는 오프닝 구간이 아니다. 애초에 존재하는 구간도 아니다.
case "prev":
int newPos = Math.Max(0, intPos - 10);
if (newPos >= intOpStart && newPos <= intOpEnd)
{
intPos = intOpEnd;
}
else
{
intPos = Math.Max(0, intPos - 10);
}
break;
코드를 고치면 다음과 같다.
0값 아래로 내려가지 않도록 한 newPos를 통해 오프닝구간과 비교를 한다.
'개발' 카테고리의 다른 글
Failed to create Android keystore. Android SDK is not found or is invalid (0) | 2025.03.20 |
---|---|
개인개발_TIL_17일차(코딩테스트 - 올바른 괄호) (0) | 2025.03.19 |