티스토리 뷰
문제 설명
카카오톡 오픈채팅방에서는 친구가 아닌 사람들과 대화를 할 수 있는데, 본래 닉네임이 아닌 가상의 닉네임을 사용하여 채팅방에 들어갈 수 있다.
신입사원인 김크루는 카카오톡 오픈 채팅방을 개설한 사람을 위해, 다양한 사람들이 들어오고, 나가는 것을 지켜볼 수 있는 관리자창을 만들기로 했다. 채팅방에 누군가 들어오면 다음 메시지가 출력된다.
"[닉네임]님이 들어왔습니다."
채팅방에서 누군가 나가면 다음 메시지가 출력된다.
"[닉네임]님이 나갔습니다."
채팅방에서 닉네임을 변경하는 방법은 다음과 같이 두 가지이다.
- 채팅방을 나간 후, 새로운 닉네임으로 다시 들어간다.
- 채팅방에서 닉네임을 변경한다.
닉네임을 변경할 때는 기존에 채팅방에 출력되어 있던 메시지의 닉네임도 전부 변경된다.
예를 들어, 채팅방에 "Muzi"와 "Prodo"라는 닉네임을 사용하는 사람이 순서대로 들어오면 채팅방에는 다음과 같이 메시지가 출력된다.
"Muzi님이 들어왔습니다."
"Prodo님이 들어왔습니다."
채팅방에 있던 사람이 나가면 채팅방에는 다음과 같이 메시지가 남는다.
"Muzi님이 들어왔습니다."
"Prodo님이 들어왔습니다."
"Muzi님이 나갔습니다."
Muzi가 나간후 다시 들어올 때, Prodo 라는 닉네임으로 들어올 경우 기존에 채팅방에 남아있던 Muzi도 Prodo로 다음과 같이 변경된다.
"Prodo님이 들어왔습니다."
"Prodo님이 들어왔습니다."
"Prodo님이 나갔습니다."
"Prodo님이 들어왔습니다."
채팅방은 중복 닉네임을 허용하기 때문에, 현재 채팅방에는 Prodo라는 닉네임을 사용하는 사람이 두 명이 있다. 이제, 채팅방에 두 번째로 들어왔던 Prodo가 Ryan으로 닉네임을 변경하면 채팅방 메시지는 다음과 같이 변경된다.
"Prodo님이 들어왔습니다."
"Ryan님이 들어왔습니다."
"Prodo님이 나갔습니다."
"Prodo님이 들어왔습니다."
채팅방에 들어오고 나가거나, 닉네임을 변경한 기록이 담긴 문자열 배열 record가 매개변수로 주어질 때, 모든 기록이 처리된 후, 최종적으로 방을 개설한 사람이 보게 되는 메시지를 문자열 배열 형태로 return 하도록 solution 함수를 완성하라.
- record는 다음과 같은 문자열이 담긴 배열이며, 길이는 1 이상 100,000 이하이다.
- 다음은 record에 담긴 문자열에 대한 설명이다.
- 모든 유저는 [유저 아이디]로 구분한다.
- [유저 아이디] 사용자가 [닉네임]으로 채팅방에 입장 - "Enter [유저 아이디] [닉네임]" (ex. "Enter uid1234 Muzi")
- [유저 아이디] 사용자가 채팅방에서 퇴장 - "Leave [유저 아이디]" (ex. "Leave uid1234")
- [유저 아이디] 사용자가 닉네임을 [닉네임]으로 변경 - "Change [유저 아이디] [닉네임]" (ex. "Change uid1234 Muzi")
- 첫 단어는 Enter, Leave, Change 중 하나이다.
- 각 단어는 공백으로 구분되어 있으며, 알파벳 대문자, 소문자, 숫자로만 이루어져있다.
- 유저 아이디와 닉네임은 알파벳 대문자, 소문자를 구별한다.
- 유저 아이디와 닉네임의 길이는 1 이상 10 이하이다.
- 채팅방에서 나간 유저가 닉네임을 변경하는 등 잘못 된 입력은 주어지지 않는다
예시
record result
["Enter uid1234 Muzi", "Enter uid4567 Prodo","Leave uid1234","Enter uid1234 Prodo","Change uid4567 Ryan"] | ["Prodo님이 들어왔습니다.", "Ryan님이 들어왔습니다.", "Prodo님이 나갔습니다.", "Prodo님이 들어왔습니다."] |
풀이
전체적인 풀이를 생각해내는데는 오래걸리지않았다.
Enter / Leave / Change를 일단 각각 인식해야 한다는 생각이 먼저 들었다. 따라서 Enter와 Leave인 경우엔 "들어왔습니다." 와 "나갔습니다." 를 이용하면 된다고 생각했다.
하지만 분명 닉네임을 알아야하기 때문에 공백으로 나눈 세 개의 값 중 id에 해당하는 값으로 "\(id)님이 들어왔습니다." 와 같이 작성하였다.
이제 남은 것은 Change이다. 어차피 나중에 id를 이름으로 치환해줘야 하기때문에 dictionary를 이용하여 각 id를 키로 사용하고 이름을 value로 넣어주었다. 이렇게 작성하면 Change일때는 updateValue로 값만 바꿔주면되기 때문에 간편할 것이라고 생각했다.
지금까지 한 것으로는 아직까지 "\(id)님이 들어오셨습니다."와 같이 id로 출력이 되는 상태이다. 이제 replacingOccurences를 이용하여 각 id를 키로 갖는 value를 dictionary에서 가져와 치환해주었다.
마지막으로 출력값이 알맞게 나오도록 처리해주었다.
import Foundation
func solution(_ record:[String]) -> [String] {
//"id님이 들어왔습니다."하고 id를 change있으면 바꾸기. dictionary로 아이디랑 닉네임
var result: [String] = []
var idDic: [String: String] = [:]
var idList: [String] = []
for i in record {
let status = i.components(separatedBy: " ")[0]
if status == "Enter" {
let id = i.components(separatedBy: " ")[1]
let name = i.components(separatedBy: " ")[2]
idList.append(id)
idDic.updateValue(name, forKey: id)
result.append("\(id)님이 들어왔습니다. \(id)")
} else if status == "Leave" {
let id = i.components(separatedBy: " ")[1]
result.append("\(id)님이 나갔습니다. \(id)")
} else {
let id = i.components(separatedBy: " ")[1]
let name = i.components(separatedBy: " ")[2]
idDic.updateValue(name, forKey: id)
}
}
for i in 0..<result.count {
let id = result[i].components(separatedBy: " ")[2]
result[i] = result[i].replacingOccurrences(of: id, with: idDic[id]!)
result[i] = result[i].components(separatedBy: " ")[0] + " " + result[i].components(separatedBy: " ")[1]
}
return result
}
결론
구현하는데에 오래걸리진않았다. 하지만 효율적이라고 확신은 못하겠다. 물론 처음에 했던 방법으로는 시간초과가 나왔기때문에 나름대로 최선을 다해서 시간복잡도를 줄인 코드이긴하다. 그래도 문자열 처리에 관련된 문제들을 더 풀어봐야 할 것 같다.
'PS' 카테고리의 다른 글
[Swift]프로그래머스_메뉴 리뉴얼 (0) | 2022.04.24 |
---|---|
[Swift]프로그래머스_타겟 넘버 (0) | 2022.04.22 |
[Swift]프로그래머스_프린터 (0) | 2022.04.19 |
[Swift]프로그래머스_기능개발 (0) | 2022.04.18 |
[Swift]프로그래머스_H-Index (0) | 2022.04.17 |
- Total
- Today
- Yesterday