[프로그래머스] 2021 카카오 블라인드 채용 > 신규 아이디 추천 (72410)(Kotlin)
문제 설명
카카오에 입사한 신입 개발자 네오
는 “카카오계정개발팀”에 배치되어, 카카오 서비스에 가입하는 유저들의 아이디를 생성하는 업무를 담당하게 되었습니다. “네오”에게 주어진 첫 업무는 새로 가입하는 유저들이 카카오 아이디 규칙에 맞지 않는 아이디를 입력했을 때, 입력된 아이디와 유사하면서 규칙에 맞는 아이디를 추천해주는 프로그램을 개발하는 것입니다.
다음은 카카오 아이디의 규칙입니다.
- 아이디의 길이는 3자 이상 15자 이하여야 합니다.
- 아이디는 알파벳 소문자, 숫자, 빼기(
-
), 밑줄(_
), 마침표(.
) 문자만 사용할 수 있습니다. - 단, 마침표(
.
)는 처음과 끝에 사용할 수 없으며 또한 연속으로 사용할 수 없습니다.
“네오”는 다음과 같이 7단계의 순차적인 처리 과정을 통해 신규 유저가 입력한 아이디가 카카오 아이디 규칙에 맞는 지 검사하고 규칙에 맞지 않은 경우 규칙에 맞는 새로운 아이디를 추천해 주려고 합니다.
신규 유저가 입력한 아이디가 new_id
라고 한다면,
1단계 new_id의 모든 대문자를 대응되는 소문자로 치환합니다.
2단계 new_id에서 알파벳 소문자, 숫자, 빼기(-), 밑줄(_), 마침표(.)를 제외한 모든 문자를 제거합니다.
3단계 new_id에서 마침표(.)가 2번 이상 연속된 부분을 하나의 마침표(.)로 치환합니다.
4단계 new_id에서 마침표(.)가 처음이나 끝에 위치한다면 제거합니다.
5단계 new_id가 빈 문자열이라면, new_id에 "a"를 대입합니다.
6단계 new_id의 길이가 16자 이상이면, new_id의 첫 15개의 문자를 제외한 나머지 문자들을 모두 제거합니다.
만약 제거 후 마침표(.)가 new_id의 끝에 위치한다면 끝에 위치한 마침표(.) 문자를 제거합니다.
7단계 new_id의 길이가 2자 이하라면, new_id의 마지막 문자를 new_id의 길이가 3이 될 때까지 반복해서 끝에 붙입니다.
예를 들어, new_id 값이 “…!@BaT#*..y.abcdefghijklm” 라면, 위 7단계를 거치고 나면 new_id는 아래와 같이 변경됩니다.
1단계 대문자 ‘B’와 ‘T’가 소문자 ‘b’와 ‘t’로 바뀌었습니다.
"...!@BaT#*..y.abcdefghijklm"
→ "...!@bat#*..y.abcdefghijklm"
2단계 ‘!’, ‘@’, ‘#’, ‘*’ 문자가 제거되었습니다.
"...!@bat#*..y.abcdefghijklm"
→ "...bat..y.abcdefghijklm"
3단계 ‘…‘와 ‘..’ 가 ‘.’로 바뀌었습니다.
"...bat..y.abcdefghijklm"
→ ".bat.y.abcdefghijklm"
4단계 아이디의 처음에 위치한 ‘.’가 제거되었습니다.
".bat.y.abcdefghijklm"
→ "bat.y.abcdefghijklm"
5단계 아이디가 빈 문자열이 아니므로 변화가 없습니다.
"bat.y.abcdefghijklm"
→ "bat.y.abcdefghijklm"
6단계 아이디의 길이가 16자 이상이므로, 처음 15자를 제외한 나머지 문자들이 제거되었습니다.
"bat.y.abcdefghijklm"
→ "bat.y.abcdefghi"
7단계 아이디의 길이가 2자 이하가 아니므로 변화가 없습니다.
"bat.y.abcdefghi"
→ "bat.y.abcdefghi"
따라서 신규 유저가 입력한 new_id가 “…!@BaT#*..y.abcdefghijklm”일 때, 네오의 프로그램이 추천하는 새로운 아이디는 “bat.y.abcdefghi” 입니다.
신규 유저가 입력한 아이디를 나타내는 new_id가 매개변수로 주어질 때, “네오”가 설계한 7단계의 처리 과정을 거친 후의 추천 아이디를 return 하도록 solution 함수를 완성해 주세요.
제한 사항
new_id는 길이 1 이상 1,000 이하인 문자열입니다.
new_id는 알파벳 대문자, 알파벳 소문자, 숫자, 특수문자로 구성되어 있습니다.
new_id에 나타날 수 있는 특수문자는 -_.~!@#$%^&*()=+[{]}:?,<>/
로 한정됩니다.
예제 입출력
no | new_id | result |
---|---|---|
예1 | "...!@BaT#*..y.abcdefghijklm" |
"bat.y.abcdefghi" |
예2 | "z-+.^." |
"z--" |
예3 | "=.=" |
"aaa" |
예4 | "123_.def" |
"123_.def" |
예5 | "abcdefghijklmn.p" |
"abcdefghijklmn" |
입출력 예 #1
문제의 예시와 같습니다.
입출력 예 #2
7단계를 거치는 동안 new_id가 변화하는 과정은 아래와 같습니다.
1단계 변화 없습니다.
2단계 "z-+.^."
→ "z-.."
3단계 "z-.."
→ "z-."
4단계 "z-."
→ "z-"
5단계 변화 없습니다.
6단계 변화 없습니다.
7단계 "z-"
→ "z--"
입출력 예 #3
7단계를 거치는 동안 new_id가 변화하는 과정은 아래와 같습니다.
1단계 변화 없습니다.
2단계 "=.="
→ "."
3단계 변화 없습니다.
4단계 "."
→ ""
(new_id가 빈 문자열이 되었습니다.)
5단계 ""
→ "a"
6단계 변화 없습니다.
7단계 "a"
→ "aaa"
입출력 예 #4
1단계에서 7단계까지 거치는 동안 new_id(“123_.def”)는 변하지 않습니다. 즉, new_id가 처음부터 카카오의 아이디 규칙에 맞습니다.
입출력 예 #5
1단계 변화 없습니다.
2단계 변화 없습니다.
3단계 변화 없습니다.
4단계 변화 없습니다.
5단계 변화 없습니다.
6단계 "abcdefghijklmn.p"
→ "abcdefghijklmn."
→ "abcdefghijklmn"
7단계 변화 없습니다.
문제 풀이1
import java.util.LinkedList
class Solution {
val queue: LinkedList<Char> = LinkedList()
fun solution(new_id: String): String {
val filtered_id = filter(new_id)
var prev = ' '
filtered_id.forEach {
if (prev == '.' && it == '.') {
return@forEach
} else {
queue.add(it)
}
prev = it
}
removeDot()
id_stratcher()
removeDot()
return queue.joinToString("")
}
fun filter(new_id: String): String {
val sb = StringBuilder()
val allowedChar = CharArray(10) { '0' + it } +
CharArray(26) { 'a' + it } +
charArrayOf('_', '-', '.')
new_id.forEach {
when (it) {
!in allowedChar -> {
if (it in 'A'..'Z') {
sb.append(it.toLowerCase())
}
}
in allowedChar -> {
sb.append(it)
}
}
}
return sb.toString()
}
fun removeDot() {
if (queue.peekFirst() == '.') queue.pollFirst()
if (queue.peekLast() == '.') queue.pollLast()
}
fun id_stratcher() {
when {
queue.isNullOrEmpty() -> {
queue.add('a')
queue.add('a')
queue.add('a')
}
queue.size > 15 -> {
while (queue.size != 15) {
queue.pollLast()
}
}
queue.size < 3 -> {
while (queue.size < 3) {
queue.add(queue.peekLast())
}
}
}
}
}
문제 풀이2
class Solution {
fun solution(new_id: String): String {
// 정답을 리턴할 String 변수 선언
var answer: String = ""
// 0~9, 'a' ~ 'z', '_', '-', '.' 의 허용 문자열 캐릭터어레이 생성
val allowedChar = CharArray(10) { '0' + it } +
CharArray(26) { 'a' + it } + charArrayOf('_', '-', '.')
// new_id에 관하여 i(인덱스: int i = 0; i < new_id.length ; i++)로 하나씩 탐색
for (i in new_id.indices) {
//new_id[i]가 allowedChar 배열의 원소라면,
//answer(String) 제일 뒤에 new_id[i](Char)를 더한 것을 answer에 저장
if (new_id[i] in allowedChar) {
answer += new_id[i]
//new_id[i]가 allowedChar 배열의 원소가 아니지만 'A'부터 'Z'까지 범위에 포함되는 문자라면,
//answer[i](String) 제일 뒤에 new_id[i](Char)의 소문자를 더한 것을 answer에 저장
//ascii코드 상에서 'a'(소문자)는 'A'(대문자)보다 순서적으로 32번째 뒤에 존재
} else if (new_id[i] in 'A'..'Z') {
answer += (new_id[i] + 32)
}
// ELSE: 허용하는 문자가 아니면 CONTINUE
}
// answer 문자열에 ".."를 "."로 모두 바뀔때까지 반복
while (answer.contains("..")) {
answer = answer.replace("..", ".")
}
// answer는 비어있는 문자열이 아니고, answer[0] == '.'이면
if (answer.isNotBlank() && answer[0] == '.') {
answer = answer.substring( 1 until answer.length)
}
// answer의 길이는 3이상이고, answer의 마지막 문자가 '.'이면
if (answer.length > 2 && answer[answer.length - 1] == '.') {
answer = answer.substring(0 until answer.length - 1)
}
do {
// answer문자열의 길이가 3 미만일 경우,
if (answer.length < 3) {
when (answer.length) {
// answer문자열이 비어있는 경우 "aaa" 반환
0 -> return "aaa"
1 -> {
//answer문자열의 길이가 1이고, 그 문자가 '.'가 아니라면
//answer[0]문자를 3번 반복해 문자열로 만든 것을 반환
return if (answer[0] != '.') answer.repeat(3)
//answer문자열의 길이가 1이고, 그 문자가 '.'라면
//"aaa"를 반환
else "aaa"
}
2 -> {
//answer문자열의 길이가 2인 일때,
//두 문자 모두다 '.'인 경우 "aaa"
return if (answer[0] == answer[1] && answer[1] == '.') "aaa"
//마지막 문자가 '.'인 경우 첫번째 문자를 3번 반복해 String으로 만든 것을 반환
else if (answer[0] == '.') "" + answer[1] + answer[1] + answer[1]
//마지막 문자가 '.'인 경우 두번째 문자를 3번 반복해 String으로 만든 것을 반환
else if (answer[1] == '.') "" + answer[0] + answer[0] + answer[0]
//두 문자중 '.'이 없는 경우 기존 문자열에 두번째 문자를 마지막에 더한 것을 반환
else answer + answer[1]
}
}
//answer문자열의 길이가 3이상일 경우,
} else {
//answer문자열의 마지막 문자가 '.'인 경우
if (answer[answer.length - 1] == '.') {
answer = answer.substring(0, answer.length - 1)
}
//answer문자열의 첫 문자가 '.'인 경우
if (answer[0] == '.') {
answer = answer.substring(1, answer.length)
}
//answer문자열의 길이가 16이상인 경우,
//0번번째 문자부터 14문자까지(15개 문자)만 사용, 15번째를 포함 이후 문자 버림
if (answer.length > 15) {
answer = answer.substring(0 until 15)
}
}
// answer의 첫 문자나 마지막 문자가 '.'인 경우 do-while 반복
} while (answer[0] == '.' || answer[answer.length - 1] == '.')
return answer
}
}