일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
- 스프링 핵심 기능
- Exception
- Servlet
- SpringBoot
- kotlin
- JPQL
- JDBC
- java
- 스프링
- pointcut
- spring
- 자바
- Thymeleaf
- db
- http
- Spring Boot
- 김영한
- 알고리즘
- 백준
- Android
- 그리디
- QueryDSL
- transaction
- Greedy
- Proxy
- AOP
- jpa
- 인프런
- springdatajpa
- 스프링 핵심 원리
- Today
- Total
개발자되기 프로젝트
[프로그래머스] 문자열 압축 본문
1. 문제
데이터 처리 전문가가 되고 싶은 "어피치"는 문자열을 압축하는 방법에 대해 공부를 하고 있습니다.
최근에 대량의 데이터 처리를 위한 간단한 비손실 압축 방법에 대해 공부를 하고 있는데,
문자열에서 같은 값이 연속해서 나타나는 것을 그 문자의 개수와 반복되는 값으로 표현하여 더 짧은 문자열로 줄여서 표현하는 알고리즘을 공부하고 있습니다.
간단한 예로 "aabbaccc"의 경우 "2a2ba3c"(문자가 반복되지 않아 한번만 나타난 경우 1은 생략함)와 같이 표현할 수 있는데, 이러한 방식은 반복되는 문자가 적은 경우 압축률이 낮다는 단점이 있습니다.
예를 들면, "abcabcdede"와 같은 문자열은 전혀 압축되지 않습니다. "
어피치"는 이러한 단점을 해결하기 위해 문자열을 1개 이상의 단위로 잘라서 압축하여 더 짧은 문자열로 표현할 수 있는지 방법을 찾아보려고 합니다.
예를 들어, "ababcdcdababcdcd"의 경우 문자를 1개 단위로 자르면 전혀 압축되지 않지만,
2개 단위로 잘라서 압축한다면 "2ab2cd2ab2cd"로 표현할 수 있습니다.
다른 방법으로 8개 단위로 잘라서 압축한다면 "2ababcdcd"로 표현할 수 있으며,
이때가 가장 짧게 압축하여 표현할 수 있는 방법입니다.
다른 예로, "abcabcdede"와 같은 경우, 문자를 2개 단위로 잘라서 압축하면
"abcabc2de"가 되지만, 3개 단위로 자른다면 "2abcdede"가 되어 3개 단위가 가장 짧은 압축 방법이 됩니다.
이때 3개 단위로 자르고 마지막에 남는 문자열은 그대로 붙여주면 됩니다.
압축할 문자열 s가 매개변수로 주어질 때, 위에 설명한 방법으로 1개 이상 단위로 문자열을 잘라 압축하여 표현한 문자열 중 가장 짧은 것의 길이를 return 하도록 solution 함수를 완성해주세요.
제한사항
- s의 길이는 1 이상 1,000 이하입니다.
- s는 알파벳 소문자로만 이루어져 있습니다.
입출력 예
s | result |
"aabbaccc" | 7 |
"ababcdcdababcdcd" | 9 |
"abcabcdede" | 8 |
"abcabcabcabcdededededede" | 14 |
"xababcdcdababcdcd" | 17 |
입출력 예에 대한 설명
입출력 예 #1
문자열을 1개 단위로 잘라 압축했을 때 가장 짧습니다.
입출력 예 #2
문자열을 8개 단위로 잘라 압축했을 때 가장 짧습니다.
입출력 예 #3
문자열을 3개 단위로 잘라 압축했을 때 가장 짧습니다.
입출력 예 #4
문자열을 2개 단위로 자르면 "abcabcabcabc6de" 가 됩니다.
문자열을 3개 단위로 자르면 "4abcdededededede" 가 됩니다.
문자열을 4개 단위로 자르면 "abcabcabcabc3dede" 가 됩니다.
문자열을 6개 단위로 자를 경우 "2abcabc2dedede"가 되며, 이때의 길이가 14로 가장 짧습니다.
입출력 예 #5
문자열은 제일 앞부터 정해진 길이만큼 잘라야 합니다.
따라서 주어진 문자열을 x / ababcdcd / ababcdcd 로 자르는 것은 불가능 합니다.
이 경우 어떻게 문자열을 잘라도 압축되지 않으므로 가장 짧은 길이는 17이 됩니다.
2. 문제 정리
- 주어진 문자는 1개씩, 2개씩.....n개씩 끊어서 반복되는 문자열이 있는지 확인하는 문제이다.
- n개의 최대값을 몇일까? 반복될 수 있다는 것은 최대 주어진 문자열 길이의 반까지. 가능.
- 짝수인 경우 길이/2 이지만 홀수인 경우 길이/2+1이므로 둘 다 포함할 수 있도록 길이/2+1의 수만큼 끊어서 본다.
- 이때 같은 문자열이 있는 경우 몇개가 있는지 확인해야 한다.
- 몇개씩 있는지 확인할 때 2자리 수 이상 넘어갈 경우 어떻게 처리해야 할지 정해야 함.
3. 코드
- 가장 바깥에 있는 반복문은 i를 늘려가며 몇개씩 끊을 지 정해주는 역할을 한다.
- 몇 개씩 끊을 지 정해지면, 해당 개수만큼 끊어서 배열로 만들어 둔다.
- 이 때 i개씩 자르고 남은 문자열도 잊지 말고 넣어주자.
- 이제 문자열이 이전 문자열과 같은지 확인한다.
- 이 전 문자열과 같은 경우 count를 늘려주고
- 같지 않은 경우에 results에 이전 문자열과 count를 넣어서 문자를 압축한다.
- 문자 압축이 완료되고 압축된 문자열의 길이가 최소인지 비교한다.
import java.util.*;
class Solution {
public int solution(String s) {
int answer = s.length();
for (int i=1; i<=s.length()/2+1; i++){
//1개식, 2개씩...차례대로 비교
StringBuilder builder = new StringBuilder();
//i자씩 끊어서 배열 만들기
int Q = s.length()/i;
ArrayList<String> arrayList = new ArrayList<>();
for (int j=0; j<i*(Q); j+=i){
arrayList.add(s.substring(j, j+i));
}
//i개씩 자르고 남은 놈들
arrayList.add(s.substring(i*Q, s.length()));
//잘라낸 문자열이 같은지 확인.
ArrayList<String> results = new ArrayList<>();
String temp = "";
int cnt = 1;
for (String value : arrayList) {
if (value.equals(temp)) {
cnt++;
} else {
if (cnt != 1) {
results.add(temp + String.valueOf(cnt));
} else {
results.add(temp);
}
temp = value;
cnt = 1;
}
}
results.add(arrayList.get(Q));
results.forEach(builder::append);
String zipString = builder.toString();
// System.out.println(zipString);
int zipCount = zipString.length();
if (zipCount < answer){
answer = zipCount;
}
}
return answer;
}
}
- 로직은 맞지만 효율이 너무 떨어진다.
- 가장 바깥의 반복문은 필수적이나, 반복무 안에 반복문이 2개가 있다. 최대한 줄여보자.
'코테준비' 카테고리의 다른 글
[프로그래머스] JadenCase 문자열 만들기 (0) | 2021.12.20 |
---|---|
[프로그래머스] 카카오프레즈 컬러링북 (0) | 2021.12.19 |
[프로그래머스] 이상한 문자 만들기 (0) | 2021.12.14 |
[프로그래머스] 시저암호 (0) | 2021.12.13 |
[프로그래머스] 문자열을 정수로 바꾸기 (0) | 2021.12.13 |