프로그래머스 코딩 기초 트레이닝 Lv.0 124문제 Java 풀이 (작성중)

프로그래머스 코딩 기초 트레이닝 Lv.0 124문제

https://school.programmers.co.kr/learn/challenges/training?order=recent&page=1
문자열 연산, 배열, 조건문, 반복문 등 기본적인 Java 문법을 익힐 수 있는 문제들입니다.
정답률 높은 순으로 풀면 성취감 느끼며 기본을 다질 수 있습니다.

출력 문제 풀이

문자열 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String a = sc.next();
    sc.close();
    
    System.out.println(a);
  }
}

a와 b 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int a = sc.nextInt();
    int b = sc.nextInt();
    sc.close();

    System.out.println("a = " + a);
    System.out.println("b = " + b);
  }
}

Scanner 사용 후에는 close 함수로 키보드 입력 스트림을 닫아 시스템 자원을 낭비하지 않는 것이 중요합니다.

문자열 반복해서 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String str = sc.next();
    int n = sc.nextInt();
    
    System.out.println(str.repeat(n));
  }
}

대소문자 바꿔서 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String a = sc.next();
    
    for (char c : a.toCharArray()) {
      System.out.print(Character.isUpperCase(c)? Character.toLowerCase(c) : Character.toUpperCase(c));
    }
  }
}

특수문자 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    System.out.print("!@#$%^&*(\\'\"<>?:;");
  }
}

java에서 백슬래시()는 이스케이프 문자로 사용되며, 다음에 오는 문자를 코드가 아닌 문자로 출력합니다.

덧셈식 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int a = sc.nextInt();
    int b = sc.nextInt();

    System.out.println(a + " + " + b + " = " + (a + b));
  }
}

아래처럼, printf 함수로도 구현할 수 있습니다.

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int a = sc.nextInt();
    int b = sc.nextInt();

    System.out.printf("%d + %d = %d", a, b, a+b);
  }
}

문자열 붙여서 출력하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String a = sc.next();
    String b = sc.next();
    
    System.out.println(a + b);
  }
}

이 문제는 아래처럼 다양한 풀이가 가능합니다.

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String a = sc.next();
    for(int i=0; i < a.length(); i++){
      System.out.println(a.charAt(i));
    }
  }
}
import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String a = sc.next();
    for(String s : a.split("")){
      System.out.println(s);
    }
  }
}

문자열 돌리기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    String a = sc.next();
    
    for (char c : a.toCharArray()) {
      System.out.println(c);
    }
  }
}

홀짝 구분하기

import java.util.Scanner;

public class Solution {
  public static void main(String[] args) {
    Scanner sc = new Scanner(System.in);
    int n = sc.nextInt();
    
    if (n % 2 == 0) {
      System.out.println(n + " is even");
    } else {
      System.out.println(n + " is odd");
    }
  }
}

연산 문제 풀이

문자열 겹쳐쓰기

class Solution {
  public String solution(String my_string, String overwrite_string, int s) {
    String[] myArr = my_string.split("");
    
    int i = 0;
    
    for (char c : overwrite_string.toCharArray()) {
      myArr[s + i] = String.valueOf(c);
      
      i++;
    }
    
    return String.join("", myArr);
  }
}

아래처럼, char 배열 두개로 푸는 방법이 split + join 보다 메모리 효율적입니다.

class Solution {
  public String solution(String my_string, String overwrite_string, int s) {
    char[] myChars = my_string.toCharArray();
    char[] overChars = overwrite_string.toCharArray();
    
    for (int i = 0; i < overChars.length; i++) {
      myChars[s + i] = overChars[i];
    }
    
    return String.valueOf(myChars);
  }
}

아래와 같이, substring을 활용하면 더 간결하게 풀 수 있습니다.

class Solution {
  public String solution(String my_string, String overwrite_string, int s) {
    String before = my_string.substring(0, s);
    String after = my_string.substring(s + overwrite_string.length(), my_string.length());
    
    return before + overwrite_string + after;
  }
}

문자열 섞기

class Solution {
  public String solution(String str1, String str2) {
    String answer = "";
    
    for (int i = 0; i < str1.length(); i++) {
      answer += str1.charAt(i) + "" + str2.charAt(i);
    }
    
    return answer;
  }
}

아래처럼, StringBuilder를 사용하여 문자열 결합하는 것이 메모리 효율적입니다.

class Solution {
  public String solution(String str1, String str2) {
    char[] chars1 = str1.toCharArray();
    char[] chars2 = str2.toCharArray();

    StringBuilder sb = new StringBuilder();
    for (int i=0; i<chars1.length; i++) {
      sb.append(chars1[i]);
      sb.append(chars2[i]);
    }

    return sb.toString();
  }
}

StringBuilder는 java.lang 패키지에 포함되어 있어, import문 없이 사용할 수 있습니다.

문자 리스트를 문자열로 변환하기

class Solution {
  public String solution(String[] arr) {
    return String.join("", arr);
  }
}

문자열 곱하기

class Solution {
  public String solution(String my_string, int k) {
    return my_string.repeat(k);
  }
}

더 크게 합치기

class Solution {
  public int solution(int a, int b) {
    return Math.max(Integer.parseInt(a + "" + b), Integer.parseInt(b + "" + a));
  }
}

두 수의 연산값 비교하기

class Solution {
  public int solution(int a, int b) {
    int case1 = Integer.parseInt(String.valueOf(a) + String.valueOf(b));
    
    int case2 = 2 * a * b;
    
    return Math.max(case1, case2);
  }
}

n의 배수

class Solution {
  public int solution(int num, int n) {
    return num % n == 0 ? 1 : 0;
  }
}

공배수

class Solution {
  public int solution(int number, int n, int m) {
    int answer = 0;
    
    if (number % n == 0 && number % m == 0) {
      answer = 1;
    }
    
    return answer;
  }
}

아래처럼, 삼항연산자를 사용하면 더 짧게 구현할 수 있습니다.

class Solution {
  public int solution(int number, int n, int m) {
    return number % n == 0 && number % m == 0 ? 1 : 0;
  }
}

조건문 문제 풀이

홀짝에 따라 다른 값 반환하기

class Solution {
  public int solution(int n) {
    int answer = 0;
    
    boolean isEven = false;
    
    if (n % 2 == 0) {
      isEven = true;
    }
    
    for (int i = 1; i <= n; i++) {
      if (isEven) {
        if (i % 2 == 0) {
          answer += (i * i);
        }
      } else {
        if (i % 2 == 1) {
          answer += i;
        }
      }
    }
    
    return answer;
  }
}

아래처럼, 홀짝 for문을 분리하면 반복 중 비교 연산을 줄일 수 있습니다.

class Solution {
  public int solution(int n) {
    int answer = 0;
    
    if (n % 2 == 0) {
      for (int i = 2; i <= n; i += 2) {
        answer += i * i;
      }
    } else {
      for (int i = 1; i <= n; i += 2) {
        answer += i;
      }
    }
    
    return answer;
  }
}

조건 문자열

class Solution {
  public int solution(String ineq, String eq, int n, int m) {
      int answer = 0;
      
      if (ineq.equals(">")) {
        if (eq.equals("=")) {
          answer = n >= m? 1 : 0;
        } else {
          answer = n > m? 1 : 0;
        }
          
      } else {
        if (eq.equals("=")) {
          answer = n <= m? 1 : 0;
        } else {
          answer = n < m? 1 : 0;
        }
          
      }
      
      return answer;
  }
}

아래처럼 switch문을 사용하면 가독성과 유지보수성이 조금 더 높아집니다.

class Solution {
  public int solution(String ineq, String eq, int n, int m) {
    boolean answer = false;
    
    switch(ineq + eq) {
      case ">=" :
        answer = n >= m;
        break;
      case ">!" :
        answer = n > m;
        break;
      case "<=" :
        answer = n <= m;
        break;
      case "<!" :
        answer = n < m;
        break;
    }
    
    return answer? 1:0;
  }
}

flag에 따라 다른 값 반환하기

class Solution {
  public int solution(int a, int b, boolean flag) {
    if (flag) {
      return a + b;
    } else {
      return a - b;
    }
  }
}

아래처럼 삼항연산자로 더 간단하게 구현할 수 있습니다.

class Solution {
  public int solution(int a, int b, boolean flag) {
    return flag ? a + b : a - b;
  }
}

코드 처리하기

class Solution {
  public String solution(String code) {
    StringBuilder sb = new StringBuilder();
    
    int mode = 0;
    
    for (int idx = 0; idx < code.length(); idx++) {
      char codeIdx = code.charAt(idx);
      
      if (mode == 0) {
        if (codeIdx != '1') {
          if (idx % 2 == 0) {
            sb.append(codeIdx);
          }
        } else {
          mode = 1;
        }
          
      } else {
        if (codeIdx != '1') {
          if (idx % 2 == 1) {
            sb.append(codeIdx);
          }
        } else {
          mode = 0;
        }
      }
    }
    
    String ret = sb.toString();
    
    return "".equals(ret)? "EMPTY" : ret;
  }
}

아래와 같이, 문제의 공통 조건을 리팩토링하면 더 간결해집니다.

class Solution {
  public String solution(String code) {
    StringBuilder sb = new StringBuilder();
    
    int mode = 0;
    
    for (int idx = 0; idx < code.length(); idx++) {
      char codeIdx = code.charAt(idx);
      
      if (codeIdx == '1') {
        mode = mode == 0? 1 : 0;
        continue;
      }
      
      if (idx % 2 == mode) {
        sb.append(codeIdx);
      }
    }
    
    return sb.length() == 0? "EMPTY" : sb.toString();
  }
}

등차수열의 특정한 항만 더하기

class Solution {
  public int solution(int a, int d, boolean[] included) {
    int answer = 0;
    
    for (int i = 0; i < included.length; i++) {
      if (included[i]) {
        answer += a + (i * d);
      }
    }
    
    return answer;
  }
}

주사위 게임 2

import java.util.*;

class Solution {
  public int solution(int a, int b, int c) {
    int answer = 0;
    
    Set<Integer> hashSet = new HashSet<>();
    
    hashSet.add(a);
    hashSet.add(b);
    hashSet.add(c);
    
    switch (hashSet.size()) {
      case 3:
        answer = a + b + c;
        break;
      case 2:
        answer = (a + b + c) * (a*a + b*b + c*c);
        break;
      case 1:
        answer = (a + b + c) * (a*a + b*b + c*c) * (a*a*a + b*b*b + c*c*c);
        break;
    }
    
    return answer;
  }
}

(int) Math.pow(a, 제곱수) 로 제곱을 구할 수 있으나, 가독성을 위해 사용하지 않았습니다.

원소들의 곱과 합

class Solution {
  public int solution(int[] num_list) {
    int gop = 1;
    int sum = 0;

    for (int num : num_list) {
      gop *= num;
      sum += num;
    }
    
    return gop <= (sum * sum) ? 1 : 0;
  }
}

이어 붙인 수

class Solution {
  public int solution(int[] num_list) {
    int answer = 0;
    String odd = "";
    String even = "";
    
    for (int num : num_list) {
      if (num % 2 == 1) {
        odd += String.valueOf(num);
      } else {
        even += String.valueOf(num);
      }
    }
    
    return Integer.parseInt(odd) + Integer.parseInt(even);
  }
}

마지막 두 원소

class Solution {
  public int[] solution(int[] num_list) {
    int[] answer = new int[num_list.length+1];
    
    for (int i = 0; i < num_list.length; i++) {
      answer[i] = num_list[i];
    }
    
    int last = answer[num_list.length-1];
    int last_prev = answer[num_list.length-2];
    
    if (last > last_prev) {
      answer[num_list.length] = last - last_prev;
    } else {
      answer[num_list.length] = last * 2;
    }
    
    return answer;
  }
}

아래처럼, 마지막 부분을 삼항 연산자로 줄일 수 있습니다.

class Solution {
  public int[] solution(int[] num_list) {
    int[] answer = new int[num_list.length+1];
    
    for (int i = 0; i < num_list.length; i++) {
      answer[i] = num_list[i];
    }
    
    int last = answer[num_list.length-1];
    int last_prev = answer[num_list.length-2];
    
    answer[num_list.length] = last > last_prev ? last - last_prev : last * 2;
    
    return answer;
  }
}

수 조작하기 1

class Solution {
  public int solution(int n, String control) {
    int answer = n;
    
    for (char c : control.toCharArray()) {
      if (c == 'w') {
        answer += 1;
      } else if (c == 's') {
        answer -= 1;
      } else if (c == 'd') {
        answer += 10;
      } else {
        answer -= 10;
      } 
    }
    
    return answer;
  }
}

수 조작하기 2

class Solution {
  public String solution(int[] numLog) {
    String answer = "";
    
    for (int i = 0; i < numLog.length-1; i++) {
      int move = numLog[i+1] - numLog[i];
      
      switch (move) {
        case 1:
          answer += "w";
          break;
        case -1:
          answer += "s";
          break;
        case 10:
          answer += "d";
          break;
        case -10:
          answer += "a";
          break;
      }
    }
    
    return answer;
  }
}

아래처럼, i를 1부터 시작해도 동일하게 구현할 수 있습니다.

class Solution {
  public String solution(int[] numLog) {
    String answer = "";
    
    for (int i = 1; i < numLog.length; i++) {
      int move = numLog[i] - numLog[i-1];
      
      switch (move) {
        case 1:
          answer += "w";
          break;
        case -1:
          answer += "s";
          break;
        case 10:
          answer += "d";
          break;
        case -10:
          answer += "a";
          break;
      }
    }
    
    return answer;
  }
}

간단한 논리 연산

class Solution {
  public boolean solution(boolean x1, boolean x2, boolean x3, boolean x4) {
    return (x1 || x2) && (x3 || x4);
  }
}

∨는 OR 연산 (합집합), ∧는 and 연산 (교집합)입니다.

주사위 게임 3

import java.util.Arrays;

class Solution {
  public int solution(int a, int b, int c, int d) {

    int[] dice = { a, b, c, d };
    Arrays.sort(dice);

    int answer = 0;

    if (dice[0] == dice[3]) {
      // 네 숫자가 같은 경우
      answer = 1111 * dice[3];
    } else if (dice[0] == dice[2] || dice[1] == dice[3]) {
      // 세 숫자가 같은 경우
      answer = (int) Math.pow(10 * dice[1] + (dice[0] + dice[3] - dice[1]), 2);
    } else if (dice[0] == dice[1] && dice[2] == dice[3]) {
      // 두 숫자 쌍이 같은 경우
      answer = (dice[0] + dice[3]) * (dice[3] - dice[0]);
    } else if (dice[0] == dice[1]) {
      // 두 숫자만 같은 경우 1
      answer = dice[2] * dice[3];
    } else if (dice[1] == dice[2]) {
      // 두 숫자만 같은 경우 2
      answer = dice[0] * dice[3];
    } else if (dice[2] == dice[3]) {
      // 두 숫자만 같은 경우 3
      answer = dice[0] * dice[1];
    } else {
      // 모든 숫자가 다른 경우
      answer = dice[0];
    }

    return answer;
  }
}

배열의 원소 삭제하기

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[] delete_list) {
    Set<Integer> delSet = new HashSet<>();
    List<Integer> list = new ArrayList<>();
    
    for (int i = 0; i < delete_list.length; i++) {
      delSet.add(delete_list[i]);
    }
    
    for (int i = 0; i < arr.length; i++) {
      if (!delSet.contains(arr[i])) {
        list.add(arr[i]);
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}
import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[] delete_list) {
    List<Integer> list = new ArrayList<>();
    
    for (int i = 0; i < arr.length; i++) {
      list.add(arr[i]);
    }
    
    for (int i = 0; i < delete_list.length; i++) {
      // 형변환 하지 않으면 int형이라서 index에 해당하는 원소 삭제하여 에러 발생
      // Integer로 형변환 하면 값에 해당하는 원소 삭제하여 정상 작동
      list.remove((Integer) delete_list[i]);
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

부분 문자열인지 확인하기

class Solution {
  public int solution(String my_string, String target) {
    return my_string.contains(target)? 1 : 0;
  }
}

부분 문자열

class Solution {
  public int solution(String str1, String str2) {
    return str2.contains(str1)? 1 : 0;
  }
}

꼬리 문자열

import java.util.*;

class Solution {
  public String solution(String[] str_list, String ex) {
    StringBuilder sb = new StringBuilder();
    
    for (String str : str_list) {
      if (!str.contains(ex)) {
        sb.append(str);
      }
    }
    
    return sb.toString();
  }
}

정수 찾기

class Solution {
  public int solution(int[] num_list, int n) {
    int answer = 0;
    
    for (int num : num_list) {
      if (num == n) {
        answer = 1;
      }
    }
    
    return answer;
  }
}

주사위 게임 1

class Solution {
  public int solution(int a, int b) {
    if (a % 2 == 1 && b % 2 == 1) {
      return a*a + b*b;
    } else if (a % 2 == 1 || b % 2 == 1) {
      return 2 * (a + b);
    } else {
      return Math.abs(a - b);
    }
  }
}

날짜 비교하기

class Solution {
  public int solution(int[] date1, int[] date2) {
    String date1Str = "";
    String date2Str = "";
    
    for (int d1 : date1) {
      date1Str += d1;
    }
    
    for (int d2 : date2) {
      date2Str += d2;
    }
    
    return Integer.parseInt(date1Str) < Integer.parseInt(date2Str) ? 1 : 0;
  }
}

java 날짜 비교는 아래처럼 LocalDate 객체를 이용하는 것이 정확합니다.

import java.time.LocalDate;

class Solution {
  public int solution(int[] date1, int[] date2) {
      
    LocalDate localDate1 = LocalDate.of(date1[0], date1[1], date1[2]);
    LocalDate localDate2 = LocalDate.of(date2[0], date2[1], date2[2]);
    
    return localDate1.isBefore(localDate2) ? 1 : 0;
  }
}

LocalDate에서 제공하는 함수는 isBefore, isAfter, isEqual 등이 있습니다.

커피 심부름

class Solution {
  public int solution(String[] order) {
    int answer = 0;
    
    for (String order_str : order) {
      if (order_str.indexOf("cafelatte") != -1) {
        answer += 5000;
      } else { 
        answer += 4500;
      }
    }
    
    return answer;
  }
}

아래처럼, contains 함수를 사용하면 더 간결하게 구현할 수 있습니다.

class Solution {
  public int solution(String[] order) {
    int answer = 0;
    
    for (String order_str : order) {
      answer += order_str.contains("cafelatte")? 5000 : 4500;
    }
        
    return answer;
  }
}

반복문 문제 풀이

수열과 구간 쿼리 3

class Solution {
  public int[] solution(int[] arr, int[][] queries) {
    for (int i = 0; i < queries.length; i++) {
      int index_i = queries[i][0];
      int index_j = queries[i][1];
      
      int temp = arr[index_i];
      arr[index_i] = arr[index_j];
      arr[index_j] = temp;
    }
    
    return arr;
  }
}

아래처럼, 향상된 for문을 사용하는 것도 좋습니다.

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[][] queries) {
    for (int[] query : queries) {
      int i = query[0];
      int j = query[1];
      
      int temp = arr[i];
      arr[i] = arr[j];
      arr[j] = temp;
    }
    
    return arr;
  }
}

수열과 구간 쿼리 2

class Solution {
  public int[] solution(int[] arr, int[][] queries) {
    int[] answer = new int[queries.length];
    
    for (int i = 0; i < queries.length; i++) {
      int min = 1000000;
      for (int j = queries[i][0]; j <= queries[i][1]; j++) {
        if (arr[j] > queries[i][2]) {
          min = Math.min(min, arr[j]);
        }
      }
      
      if (min == 1000000) {
        min = -1;
      }
      
      answer[i] = min;
    }
    
    return answer;
  }
}

아래와 같이, Arrays.fill 함수를 사용하여 배열을 -1로 채우고 삼항연산자를 이용하면,
불필요한 min 변수를 선언하지 않아서 더 간결하고 효율적입니다.

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[][] queries) {
    int[] answer = new int[queries.length];
    Arrays.fill(answer, -1);
    
    for (int i = 0; i < queries.length; i++) {
      for (int j = queries[i][0]; j <= queries[i][1]; j++) {
        if (arr[j] > queries[i][2]) {
          answer[i] = answer[i] == -1? arr[j] : Math.min(answer[i], arr[j]);
        }
      }
    }
    
    return answer;
  }
}

수열과 구간 쿼리 4

class Solution {
  public int[] solution(int[] arr, int[][] queries) {
    for (int[] query : queries) {
      int s = query[0];
      int e = query[1];
      int k = query[2];
      
      for (int i = s; i <= e; i++) {
        if (i % k == 0) {
          arr[i]++;
        }
      }
    }
    
    return arr;
  }
}

배열 만들기 2

import java.util.*;

class Solution {
  public int[] solution(int l, int r) {
    List<Integer> list = new ArrayList<>();
    
    for (int i = l; i <= r; i++) {
      char[] charArr = String.valueOf(i).toCharArray();
      
      boolean isAnswer = true;
      
      for (int j = 0; j < charArr.length; j++) {
        if (charArr[j] != '0' && charArr[j] != '5') {
          isAnswer = false;
          break;
        }
      }
      
      if (isAnswer) {
        list.add(Integer.parseInt(String.valueOf(charArr)));
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer.length == 0 ? new int[] {-1} : answer;
  }
}

아래와 같이, 문자열 변환 없이 int 연산을 활용하면 더 메모리 효율적입니다.

import java.util.*;

class Solution {
  public int[] solution(int l, int r) {
    List<Integer> list = new ArrayList<>();
    
    for (int i = l; i <= r; i++) {
      int num = i;
      
      boolean isAnswer = true;

      while (num > 0) {
        int digit = num % 10; // 마지막 자리 확인
        if (digit != 0 && digit != 5) {
          isAnswer = false;
          break;
        }
        num /= 10; // 자리수 이동
      }
      
      if (isAnswer) {
        list.add(i);
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer.length == 0 ? new int[] {-1} : answer;
  }
}

카운트 업

class Solution {
  public int[] solution(int start_num, int end_num) {
    int[] answer = new int[end_num - start_num +1];
    
    int index = 0;
    
    for (int i = start_num; i <= end_num; i++) {
      answer[index++] = i;
    }
        
    return answer;
  }
}
class Solution {
  public int[] solution(int start_num, int end_num) {
    int[] answer = new int[end_num - start_num +1];
    
    for (int i = 0; i <= end_num - start_num; i++) {
      answer[i] = i + start_num;
    }
        
    return answer;
  }
}

콜라츠 수열 만들기

import java.util.*;

class Solution {
  public int[] solution(int n) {
    ArrayList<Integer> list = new ArrayList<>();
    
    list.add(n);
    
    while (n > 1) {
      if (n % 2 == 0) {
        n /= 2;
      } else {
        n = 3 * n + 1;
      }
      
      list.add(n);
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

배열 만들기 4

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    Deque<Integer> stack = new ArrayDeque<>();
    
    for (int i = 0; i < arr.length; i++) {
      if (stack.isEmpty()) {
        stack.push(arr[i]);
      } else {
        if(stack.peek() < arr[i]) {
          stack.push(arr[i]);
        } else {
          stack.pop();
          i -= 1;
        }
      }
    }
    
    int[] stk = new int[stack.size()];
    
    for (int i = stack.size() -1; i >= 0; i--) {
      stk[i] = stack.pop();
    }
    
    return stk;
  }
}

마지막 원소만 이용하는 문제이므로, 스택을 이용하며 풀면 효율적입니다.

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    Deque<Integer> stack = new ArrayDeque<>();
    
    int i = 0;
    
    while(i < arr.length) {
      if (stack.isEmpty() || stack.peek() < arr[i]) {
        stack.push(arr[i]);
        i += 1;
      } else {
        stack.pop();
      }
    }
    
    int[] stk = new int[stack.size()];
    
    for (int j = stack.size() -1; j >= 0; j--) {
      stk[j] = stack.pop();
    }
    
    return stk;
  }
}

위와 같이, while문을 사용하여 불필요한 i -= 1 연산을 없애고 동일한 작업을 하는 조건문을 통일하면 더 좋습니다.

그림 확대

class Solution {
  public String[] solution(String[] picture, int k) {
    String[] answer = new String[picture.length * k];
    
    for (int i = 0; i < picture.length * k; i++) {
      StringBuilder sb = new StringBuilder();
      
      for (int j = 0; j < picture[0].length() * k; j++) {
        sb.append(picture[(i / k)].charAt((j / k)));
      }
      answer[i] = sb.toString();
    }
    
    return answer;
  }
}

조건에 맞게 수열 변환하기 3

class Solution {
  public int[] solution(int[] arr, int k) {
    int[] answer = new int[arr.length];
    
    for (int i = 0; i < arr.length; i++) {
      if (k % 2 == 1) {
        answer[i] = arr[i] * k;
      } else {
        answer[i] = arr[i] + k;
      }
    }
    
    return answer;
  }
}

아래처럼, 한 번만 비교하고 각각의 for문으로 돌리는 것이 더 효율적입니다.

class Solution {
  public int[] solution(int[] arr, int k) {
    if (k % 2 == 0) {
      for (int i = 0; i < arr.length; i++) {
        arr[i] = arr[i] + k;
      }
    } else {
      for (int i = 0; i < arr.length; i++) {
        arr[i] = arr[i] * k;
      }
    }
    
    return arr;
  }
}

l로 만들기

import java.util.*;

class Solution {
  public String solution(String myString) {
    StringBuilder sb = new StringBuilder();
    
    for (char c : myString.toCharArray()) {
      if (c - 'l' > 0) {
        sb.append(c);
      } else {
        sb.append('l');
      }
    }
    
    return sb.toString();
  }
}

아래처럼, 정규표현식을 이용하면 더 간단하게 변경할 수 있습니다.

import java.util.*;

class Solution {
  public String solution(String myString) {
    return myString.replaceAll("[^l-z]", "l");
  }
}

문자열 문제 풀이

글자 이어 붙여 문자열 만들기

import java.util.*;

class Solution {
  public String solution(String my_string, int[] index_list) {
    StringBuilder sb = new StringBuilder();
    
    for (int i = 0; i < index_list.length; i++) {
      sb.append(my_string.substring(index_list[i], index_list[i]+1));
    }
    
    return sb.toString();
  }
}

아래처럼 charAt으로 문자를 추출하면 더 쉽게 이어붙일 수 있습니다.

import java.util.*;

class Solution {
  public String solution(String my_string, int[] index_list) {
    StringBuilder sb = new StringBuilder();
    
    for (int index : index_list) {
      sb.append(my_string.charAt(index));
    }
    
    return sb.toString();
  }
}

9로 나눈 나머지

class Solution {
  public int solution(String number) {
    int answer = 0;
    
    for (char c : number.toCharArray()) {
      answer += c - '0';
    }
    
    return answer % 9;
  }
}

문자열 여러 번 뒤집기

class Solution {
  public String solution(String my_string, int[][] queries) {
      
    for (int i = 0; i < queries.length; i++) {
      String startStr = my_string.substring(0, queries[i][0]);
      
      String middleStr = my_string.substring(queries[i][0], queries[i][1]+1);
      
      StringBuilder sb = new StringBuilder();
      
      for (int j = middleStr.length()-1; j >= 0; j--) {
        sb.append(middleStr.charAt(j));
      }
      
      String endStr = my_string.substring(queries[i][1]+1, my_string.length());

      my_string = startStr + sb.toString() + endStr;
    }
    
    return my_string;
  }
}

아래와 같이, StringBuilder 생성 시 파라미터로 문자열을 넣고 reverse 명령어를 사용하면 더 효율적입니다.

class Solution {
  public String solution(String my_string, int[][] queries) {
      
    for (int i = 0; i < queries.length; i++) {
      StringBuilder sb = new StringBuilder(my_string.substring(queries[i][0], queries[i][1]+1));
      sb.reverse();
      
      my_string = my_string.substring(0, queries[i][0])
                + sb.toString()
                + my_string.substring(queries[i][1]+1, my_string.length());
    }
    
    return my_string;
  }
}

배열 만들기 5

import java.util.*;

class Solution {
  public int[] solution(String[] intStrs, int k, int s, int l) {
    ArrayList<Integer> list = new ArrayList<>();
    
    for (int i = 0; i < intStrs.length; i++) {
      int subInt = Integer.parseInt(intStrs[i].substring(s, s+l));
      
      if (subInt > k) {
        list.add(subInt);
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

부분 문자열 이어 붙여 문자열 만들기

class Solution {
  public String solution(String[] my_strings, int[][] parts) {
    String answer = "";
    
    for (int i = 0; i < my_strings.length; i++) {
      answer += my_strings[i].substring(parts[i][0], parts[i][1]+1);
    }
    
    return answer;
  }
}

문자열의 뒤의 n글자

class Solution {
  public String solution(String my_string, int n) {
    return my_string.substring(my_string.length()-n);
  }
}

접미사 배열

import java.util.*;

class Solution {
  public String[] solution(String my_string) {
    ArrayList<String> list = new ArrayList<>();
    
    for (int i = 0; i < my_string.length(); i++) {
      list.add( my_string.substring(i) );
    }
    
    Collections.sort(list);
    
    String[] answer = new String[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

이 문제는 정답 배열 사이즈가 정해져있기 때문에, ArrayList를 사용하지 않아도 풀 수 있습니다.

import java.util.*;

class Solution {
  public String[] solution(String my_string) {
    String[] answer = new String[my_string.length()];
    
    for (int i = 0; i < my_string.length(); i++) {
      answer[i] = my_string.substring(i);
    }
    
    Arrays.sort(answer);
    
    return answer;
  }
}

접미사인지 확인하기

class Solution {
  public int solution(String my_string, String is_suffix) {
    int i = my_string.indexOf(is_suffix, my_string.length() - is_suffix.length());
    
    if (my_string.length() < is_suffix.length()) {
      return 0;
    }
    
    return i + is_suffix.length() == my_string.length()? 1 : 0;
  }
}

아래처럼, endsWith 함수를 사용하면 접미사인지 더 쉽게 확인할 수 있습니다.

class Solution {
  public int solution(String my_string, String is_suffix) {
    if (my_string.endsWith(is_suffix)) {
      return 1;
    }
    
    return 0;
  }
}

문자열의 앞의 n글자

class Solution {
  public String solution(String my_string, int n) {
    return my_string.substring(0, n);
  }
}

접두사인지 확인하기

class Solution {
  public int solution(String my_string, String is_prefix) {
    return my_string.indexOf(is_prefix) == 0? 1 : 0;
  }
}

startsWith 함수로 더 간단히 확인할 수 있습니다.

class Solution {
  public int solution(String my_string, String is_prefix) {
    return my_string.startsWith(is_prefix)? 1 : 0;
  }
}

문자열 뒤집기

class Solution {
  public String solution(String my_string, int s, int e) {
    String[] myArr = my_string.split("");
    
    for (int i = 0; i <= e - s; i++) {
      myArr[s + i] = String.valueOf(my_string.charAt(e - i));
    }
    
    return String.join("", myArr);
  }
}

StringBuilder의 reverse 함수로 문자열을 뒤집으면 메모리와 성능 면에서 더 효율적입니다.

class Solution {
  public String solution(String my_string, int s, int e) {
    StringBuilder sb = new StringBuilder(my_string.substring(s, e+1));
    
    sb.reverse();
    
    return my_string.substring(0, s) + sb.toString() + my_string.substring(e+1);
  }
}

StringBuilder 생성자에 String을 넘기면, append 함수와 같은 효과입니다.

세로 읽기

class Solution {
  public String solution(String my_string, int m, int c) {
    String answer = "";
    
    for (int i = 0; i < my_string.length() / m; i++) {
      answer += my_string.charAt((i * m) + c -1);
    }
    
    return answer;
  }
}

아래처럼, i의 시작을 c 인덱스부터 하면 더 간결합니다.

class Solution {
  public String solution(String my_string, int m, int c) {
    String answer = "";
    
    for (int i = c - 1; i < my_string.length(); i += m) {
      answer += my_string.charAt(i);
    }
    return answer;
  }
}

qr code

class Solution {
  public String solution(int q, int r, String code) {
    String answer = "";
    
    for (int i = r; i < code.length(); i += q) {
      answer += code.charAt(i);
    }
    
    return answer;
  }
}

아래와 같이, StringBuilder를 통한 문자열 연산이 더 성능상 좋습니다.

import java.util.*;

class Solution {
  public String solution(int q, int r, String code) {
    StringBuilder sb = new StringBuilder();
    
    for (int i = r; i < code.length(); i += q) {
      sb.append(code.charAt(i));
    }
    
    return sb.toString();
  }
}

원하는 문자열 찾기

class Solution {
  public int solution(String myString, String pat) {
    return myString.toLowerCase().contains(pat.toLowerCase())? 1 : 0;
  }
}

contains 함수 뿐 아니라, indexOf 함수로도 문자열 포함 여부를 알 수 있습니다.

class Solution {
  public int solution(String myString, String pat) {
    return myString.toLowerCase().indexOf(pat.toLowerCase()) != -1? 1 : 0;
  }
}

대문자로 바꾸기

class Solution {
  public String solution(String myString) {
    return myString.toUpperCase();
  }
}

소문자로 바꾸기

class Solution {
  public String solution(String myString) {
    return myString.toLowerCase();
  }
}

배열에서 문자열 대소문자 변환하기

class Solution {
  public String[] solution(String[] strArr) {
    for (int i = 0; i < strArr.length; i++) {
      if (i % 2 == 1) {
        strArr[i] = strArr[i].toUpperCase();
      } else {
        strArr[i] = strArr[i].toLowerCase();
      }
    }
    
    return strArr;
  }
}

A 강조하기

class Solution {
  public String solution(String myString) {
    return myString.toLowerCase().replaceAll("a", "A");
  }
}

특정한 문자를 대문자로 바꾸기

class Solution {
  public String solution(String my_string, String alp) {
    return my_string.replaceAll(alp, alp.toUpperCase());
  }
}

특정 문자열로 끝나는 가장 긴 부분 문자열 찾기

class Solution {
  public String solution(String myString, String pat) {
    String answer = "";
    
    for (int i = myString.length(); i >= 0; i--) {
      String subStr = myString.substring(0, i);
      
      if (subStr.endsWith(pat)) {
        answer = subStr;
        break;
      }
    }
    
    return answer;
  }
}

뒤에서부터 문자 Index를 찾는 lastIndexOf 함수를 사용하면 더 효율적입니다.

class Solution {
  public String solution(String myString, String pat) {
    int e = myString.lastIndexOf(pat);
    
    return myString.substring(0, e + pat.length());
  }
}

문자열이 몇 번 등장하는지 세기

class Solution {
  public int solution(String myString, String pat) {
    int answer = 0;
        
    for (int i = 0; i + pat.length() <= myString.length(); i++) {
      String subString = myString.substring(i, i + pat.length());
      
      if (subString.equals(pat)) {
        answer++;
      }
    }
        
    return answer;
  }
}

아래처럼, startsWith를 사용하면 문자열로 시작 여부를 더 쉽게 확인할 수 있습니다.

class Solution {
  public int solution(String myString, String pat) {
    int answer = 0;
    
    for (int i = 0; i < myString.length(); i++) {
      if (myString.substring(i).startsWith(pat)) {
        answer++;
      }
    }
    
    return answer;
  }
}

ad 제거하기

import java.util.*;

class Solution {
  public String[] solution(String[] strArr) {
    ArrayList<String> list = new ArrayList<>();
    
    for (String str : strArr) {
      if (!str.contains("ad")) {
        list.add(str);
      }
    }
    
    String[] answer = new String[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

공백으로 구분하기 1

class Solution {
  public String[] solution(String my_string) {
    return my_string.split(" ");
  }
}

공백으로 구분하기 2

import java.util.*;

class Solution {
  public String[] solution(String my_string) {
    String[] my_arr = my_string.split(" ");
    List<String> list = new ArrayList<>();
    
    for (String str : my_arr) {
      if (!str.isEmpty()) {
        list.add(str);
      }
    }
    
    String[] answer = new String[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

아래처럼, 좌우 공백을 trim으로 제거하고 정규표현식으로 하나 이상의 공백을 자르면 좋습니다.

class Solution {
  public String[] solution(String my_string) {
    return my_string.trim().split("[ ]+");
  }
}

x 사이의 개수

import java.util.*;

class Solution {
  public int[] solution(String myString) {
    myString = myString + " ";
    
    String[] myArr = myString.split("x");
    int[] answer = new int[myArr.length];
    
    for (int i = 0; i < myArr.length; i++) {
      answer[i] = myArr[i].trim().length();
    }
    
    return answer;
  }
}

문자열 잘라서 정렬하기

import java.util.*;

class Solution {
  public String[] solution(String myString) {
    String[] myArr = myString.split("x");
    
    Arrays.sort(myArr);
    
    ArrayList<String> list = new ArrayList<>();
    
    for (String myStr : myArr) {
      if (!"".equals(myStr)) {
        list.add(myStr);
      }
    }
    
    String[] answer = new String[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

간단한 식 계산하기

class Solution {
  public int solution(String binomial) {
    int answer = 0;
    
    String[] strArr = binomial.split(" ");
    
    if (strArr[1].equals("+")) {
      answer = Integer.parseInt(strArr[0]) + Integer.parseInt(strArr[2]);
    } else if (strArr[1].equals("-")) {
      answer = Integer.parseInt(strArr[0]) - Integer.parseInt(strArr[2]);
    } else {
      answer = Integer.parseInt(strArr[0]) * Integer.parseInt(strArr[2]);
    }
    
    return answer;
  }
}

문자열 바꿔서 찾기

import java.util.*;

class Solution {
  public int solution(String myString, String pat) {
    StringBuilder sb = new StringBuilder();
    
    for (char c : myString.toCharArray()) {
      if (c == 'A') {
        sb.append('B');
      } else {
        sb.append('A');
      }
    }
    
    myString = sb.toString();
    
    return myString.contains(pat)? 1 : 0;
  }
}

이래처럼, replace를 이용하면 더 간결하게 A와 B를 바꿀 수 있습니다.

class Solution {
  public int solution(String myString, String pat) {
    myString = myString.replace("A", "a").replace("B", "A").replace("a", "B");
    return myString.contains(pat)? 1 : 0;
  }
}

rny_string

class Solution {
  public String solution(String rny_string) {
    return rny_string.replaceAll("m", "rn");
  }
}

세 개의 구분자

class Solution {
  public String[] solution(String myStr) {
    String[] answer = myStr.replaceAll("[a|b|c]", " ").trim().split("\\s+");
    
    if (answer[0].isEmpty()) {
      answer[0] = "EMPTY";
    }
    
    return answer;
  }
}

조건 문자열을 정규표현식으로 공백 치환하고, trim으로 좌우 공백을 제거합니다.
그리고 split(“\s+”)으로 1개 이상의 공백을 기준으로 잘라서 배열을 만드는 풀이입니다.

class Solution {
  public String[] solution(String myStr) {
    String[] answer = myStr.replaceAll("[a|b|c]+", " ").trim().split(" ");
    
    if (answer[0].isEmpty()) {
      answer[0] = "EMPTY";
    }
    
    return answer;
  }
}

아니면, 1개 이상의 문자열을 공백 1개로 치환하는 방법도 있습니다.

배열 문제 풀이

문자 개수 세기

class Solution {
  public int[] solution(String my_string) {
    int[] answer = new int[52];
    
    for (char c : my_string.toCharArray()) {
      if (Character.isUpperCase(c)) {
        answer[c - 'A']++;
      } else {
        answer[26 + (c - 'a')]++;
      }
    }
    
    return answer;
  }
}

배열 만들기 1

class Solution {
  public int[] solution(int n, int k) {
    int[] answer = new int[n / k];
    
    for (int i = 1; i <= n; i++) {
      if (i % k == 0) {
        answer[i / k -1] = i;
      }
    }
    
    return answer;
  }
}

글자 지우기

import java.util.*;

class Solution {
  public String solution(String my_string, int[] indices) {
    Arrays.sort(indices);
    
    for (int i = indices.length -1; i >= 0; i--) {
      my_string = my_string.substring(0, indices[i]) + my_string.substring(indices[i]+1);
    }
    
    return my_string;
  }
}

아래처럼, split과 join 함수를 활용하면 문자열을 쉽게 지울 수 있습니다.

class Solution {
  public String solution(String my_string, int[] indices) {
    String[] myArr = my_string.split("");
    
    for (int i = 0; i < indices.length; i++) {
      myArr[indices[i]] = "";
    }
    
    return String.join("", myArr);
  }
}

카운트 다운

class Solution {
  public int[] solution(int start, int end_num) {
    int[] answer = new int[start - end_num +1];
    
    for (int i = start; i >= end_num; i--) {
      answer[start - i] = i;
    }
    
    return answer;
  }
}

가까운 1 찾기

class Solution {
  public int solution(int[] arr, int idx) {
    int answer = -1;
    
    for (int i = idx; i < arr.length; i++) {
      if (arr[i] == 1) {
        return i;
      }
    }
    return answer;
  }
}

위 코드처럼 바로 return 하지 않으면, break문이 필요합니다.

class Solution {
  public int solution(int[] arr, int idx) {
    int answer = -1;
    
    for (int i = idx; i < arr.length; i++) {
      if (arr[i] == 1) {
        answer = i;
        break;
      }
    }
    return answer;
  }
}

리스트 자르기

import java.util.*;

class Solution {
  public int[] solution(int n, int[] slicer, int[] num_list) {
    int[] answer = {};
    
    int a = slicer[0];
    int b = slicer[1];
    int c = slicer[2];
    
    switch (n) {
      case 1:
        answer = Arrays.copyOfRange(num_list, 0, b+1);
        break;
      case 2:
        answer = Arrays.copyOfRange(num_list, a, num_list.length);
        break;
      case 3:
        answer = Arrays.copyOfRange(num_list, a, b+1);
        break;
      case 4:
        int[] tempArr = Arrays.copyOfRange(num_list, a, b+1);
        int[] newArr = new int[(tempArr.length+c-1) / c];
        
        for (int i = 0; i < tempArr.length; i += c) {
          newArr[i / c] = tempArr[i];
        }
        
        answer = newArr;
        
        break;
    }
    
    return answer;
  }
}

아래 코드는 위 코드와 같은 결과가 나옵니다.

import java.util.*;

class Solution {
  public int[] solution(int n, int[] slicer, int[] num_list) {
    int[] answer = {};
    
    int a = slicer[0];
    int b = slicer[1];
    int c = slicer[2];
    
    switch (n) {
      case 1:
        answer = Arrays.copyOfRange(num_list, 0, b+1);
        break;
      case 2:
        answer = Arrays.copyOfRange(num_list, a, num_list.length);
        break;
      case 3:
        answer = Arrays.copyOfRange(num_list, a, b+1);
        break;
      case 4:
        int[] tempArr = Arrays.copyOfRange(num_list, a, b+1);
        int[] newArr = new int[(b - a + c) / c];
        
        for (int i = 0, j = 0; i < tempArr.length; i += c) {
          newArr[j++] = tempArr[i];
        }
        
        answer = newArr;
        
        break;
    }
    
    return answer;
  }
}

첫 번째로 나오는 음수

class Solution {
  public int solution(int[] num_list) {
    int answer = -1;
    
    for (int i = 0; i < num_list.length; i++) {
      if (num_list[i] < 0) {
        answer = i;
        break;
      }
    }
    
    return answer;
  }
}

배열 만들기 3

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[][] intervals) {
    int[] arr1 = Arrays.copyOfRange(arr, intervals[0][0], intervals[0][1]+1);
    int[] arr2 = Arrays.copyOfRange(arr, intervals[1][0], intervals[1][1]+1);
    
    int[] answer = new int[arr1.length + arr2.length];
    
    int idx = 0;
    
    for (int i = 0; i < arr1.length; i++) {
      answer[idx++] = arr1[i];
    }
    
    for (int i = 0; i < arr2.length; i++) {
      answer[idx++] = arr2[i];
    }
    
    return answer;
  }
}

intervals[0] 인덱스 구간 arr1 배열과 intervals[1] 인덱스 구간 arr2 배열을 이어붙이는 문제입니다.
어차피 배열을 이어붙이기 위해 for문을 사용하므로, copyOfRange 함수는 필요하지 않습니다.

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[][] intervals) {
    int s1 = intervals[0][0];
    int e1 = intervals[0][1];
    int s2 = intervals[1][0];
    int e2 = intervals[1][1];
    
    int[] answer = new int[(e1 - s1 +1) + (e2 - s2 +1)];
    
    int idx = 0;
    
    for (int i = s1; i <= e1; i++) {
      answer[idx++] = arr[i];
    }
    
    for (int i = s2; i <= e2; i++) {
      answer[idx++] = arr[i];
    }
    
    return answer;
  }
}

2의 영역

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    int start_i = -1;
    int end_i = -1;
    
    for (int i = 0; i < arr.length; i++) {
      if (arr[i] == 2) {
        start_i = i;
        break;
      }
    }
    
    for (int i = arr.length-1; i >= 0; i--) {
      if (arr[i] == 2) {
        end_i = i;
        break;
      }
    }
    
    if (start_i == -1) {
      return new int[] { -1 };
    }
    
    return Arrays.copyOfRange(arr, start_i, end_i+1);
  }
}

배열 조각하기

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int[] query) {
    for (int i = 0; i < query.length; i++) {
      if (i % 2 == 0) {
        arr = Arrays.copyOfRange(arr, 0, query[i] + 1);
      } else {
        arr = Arrays.copyOfRange(arr, query[i], arr.length);
      }
    }
    
    return arr;
  }
}

n 번째 원소부터

class Solution {
  public int[] solution(int[] num_list, int n) {
    int[] answer = new int[num_list.length -n +1];
    
    for (int i = n; i <= num_list.length; i++) {
      answer[i-n] = num_list[i-1];
    }
    
    return answer;
  }
}

아래처럼, copyOfRange를 사용하면 배열을 쉽게 자를 수 있습니다.

import java.util.*;

class Solution {
  public int[] solution(int[] num_list, int n) {
    int[] answer = Arrays.copyOfRange(num_list, n-1, num_list.length);
    
    return answer;
  }
}

순서 바꾸기

import java.util.*;

class Solution {
  public int[] solution(int[] num_list, int n) {
    int[] array1 = Arrays.copyOfRange(num_list, n, num_list.length);
    int[] array2 = Arrays.copyOfRange(num_list, 0, n);
    int[] answer = new int[array1.length + array2.length];
    
    for (int i = 0; i < array1.length; i++) {
      answer[i] = array1[i];
    }
    
    for (int i = 0; i < array2.length; i++) {
      answer[array1.length + i] = array2[i];
    }
    
    return answer;
  }
}

아래처럼, 인덱스를 하나로 공유해서 쓰는 방법이 더 효율적입니다.

class Solution {
  public int[] solution(int[] num_list, int n) {
    int[] answer = new int[num_list.length];
    int idx = 0;
    
    for (int i = n; i < num_list.length; i++) {
      answer[idx++] = num_list[i];
    }
    
    for (int i = 0; i < n; i++) {
      answer[idx++] = num_list[i];
    }
    
    return answer;
  }
}

왼쪽 오른쪽

import java.util.*;

class Solution {
  public String[] solution(String[] str_list) {
    String[] answer = {};
    
    for (int i = 0; i < str_list.length; i++) {
      String str = str_list[i];
      
      if ("l".equals(str)) {
        answer = Arrays.copyOfRange(str_list, 0, i);
        break;
      } else if ("r".equals(str)) {
        answer = Arrays.copyOfRange(str_list, i+1, str_list.length);
        break;
      }
    }
    
    return answer;
  }
}

아래와 같이, 정답을 바로 return 해주는 방식도 가능합니다.

import java.util.*;

class Solution {
  public String[] solution(String[] str_list) {
      
    for (int i = 0; i < str_list.length; i++) {
      String str = str_list[i];
      
      if ("l".equals(str)) {
        return Arrays.copyOfRange(str_list, 0, i);
      } else if ("r".equals(str)) {
        return Arrays.copyOfRange(str_list, i+1, str_list.length);
      }
    }
    
    return new String[]{};
  }
}

n번째 원소까지

import java.util.*;

class Solution {
  public int[] solution(int[] num_list, int n) {
    int[] answer = Arrays.copyOfRange(num_list, 0, n);
    return answer;
  }
}

n개 간격의 원소들

import java.util.ArrayList;

class Solution {
  public int[] solution(int[] num_list, int n) {
      
    ArrayList<Integer> list = new ArrayList<>();
    
    for (int i = 0; i < num_list.length; i++) {
      if (i % n == 0) {
        list.add(num_list[i]);
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
        
    return answer;
  }
}

홀수 vs 짝수

class Solution {
  public int solution(int[] num_list) {
    int even = 0;
    int odd = 0;
    
    for (int i = 0; i < num_list.length; i++) {
      int num = num_list[i];
      
      if (i % 2 == 0) {
        even += num;
      } else {
        odd += num;
      }
    }
    
    return Math.max(even, odd);
  }
}

5명씩

class Solution {
  public String[] solution(String[] names) {
    String[] answer = new String[(names.length + 4) / 5];
    int idx = 0;
    
    for (int i = 0; i < names.length; i += 5) {
      answer[idx++] = names[i];
    }
    
    return answer;
  }
}

아래처럼, i*5번째 값을 이용하면 인덱스 하나로 구현 가능합니다.

class Solution {
  public String[] solution(String[] names) {
    String[] answer = new String[((names.length-1) / 5) +1];
    
    for (int i = 0; i < answer.length; i++) {
        answer[i] = names[i * 5];
    }
    
    return answer;
  }
}

할 일 목록

import java.util.*;

class Solution {
  public String[] solution(String[] todo_list, boolean[] finished) {
    List<String> list = new ArrayList<>();
    
    for (int i = 0; i < todo_list.length; i++) {
      if (!finished[i]) {
        list.add(todo_list[i]);
      }
    }
    
    String[] answer = new String[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

아래처럼, ArrayList를 쓰지 않고 문자열 더하기 후 split으로 구현하는 방법도 있습니다.

class Solution {
  public String[] solution(String[] todo_list, boolean[] finished) {
    String answer = new String();
    
    for (int i = 0; i < todo_list.length; i++) {
      if (!finished[i]) {
        answer += todo_list[i] + ",";
      }
    }
    
    return answer.split(",");
  }
}

n보다 커질 때까지 더하기

class Solution {
  public int solution(int[] numbers, int n) {
    int answer = 0;
    
    for (int v : numbers) {
      answer += v;
      
      if (answer > n) {
        break;
      }
    }
    
    return answer;
  }
}

수열과 구간 쿼리 1

class Solution {
  public int[] solution(int[] arr, int[][] queries) {
    for (int i = 0; i < queries.length; i++) {
      for (int s = queries[i][0]; s <= queries[i][1]; s++) {
        arr[s]++;
      }
    }
    
    return arr;
  }
}

조건에 맞게 수열 변환하기 1

class Solution {
  public int[] solution(int[] arr) {
    int[] answer = {};
    
    for (int i = 0; i < arr.length; i++) {
      int v = arr[i];
      
      if (v >= 50 && v % 2 == 0) {
        arr[i] = v / 2;
      }
          
      if (v < 50 && v % 2 == 1) {
        arr[i] = v * 2;
      }
    }
    
    return arr;
  }
}

조건에 맞게 수열 변환하기 2

import java.util.*;

class Solution {
  public int solution(int[] arr) {
    int x = 0;
    
    while(true) {
      int[] tempArr = arr.clone();
      
      // 문제 조건 반복
      for (int i = 0; i < tempArr.length; i++) {
        int value = tempArr[i];
        
        if (value >= 50 && value % 2 == 0) {
          tempArr[i] = value / 2;
        } else if (value < 50 && value % 2 == 1) {
          tempArr[i] = (value * 2) + 1;
        }
      }
      
      if (Arrays.equals(arr, tempArr)) {
        return x;
      } else {
        arr = tempArr;
      }
      
      x += 1;
    }
      
  }
}

배열 복제 시 clone 함수를 사용하지 않으면, tempArr 수정 시 원본 arr가 수정되는 버그가 생깁니다.
배열 비교 시에는 Arrays.equals 함수를 사용해야 배열 내 값까지 비교됩니다.

1로 만들기

class Solution {
  public int solution(int[] num_list) {
    int answer = 0;
    
    for (int i = 0; i < num_list.length; i++) {
      int num = num_list[i];
      
      while (num > 1) {
        if (num % 2 == 0) {
          num = num / 2;
        } else {
          num = (num - 1) / 2;
        }
        
        answer++;
      }
    }
    
    return answer;
  }
}

홀수에서 1을 빼고 2로 나누는 것, 그냥 2로 나누는 것은 결과가 동일합니다.

class Solution {
  public int solution(int[] num_list) {
    int answer = 0;
    
    for (int i = 0; i < num_list.length; i++) {
      while (num_list[i] > 1) {
        num_list[i] /= 2;
        
        answer++;
      }
    }
    
    return answer;
  }
}

길이에 따른 연산

class Solution {
  public int solution(int[] num_list) {
    int sum = 0;
    int gob = 1;
    
    for (int i = 0; i < num_list.length; i++) {
      sum += num_list[i];
      gob *= num_list[i];
    }
    
    return num_list.length > 10? sum : gob;
  }
}

배열의 원소만큼 추가하기

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    ArrayList<Integer> list = new ArrayList<>();
    
    for (int v : arr) {
      for (int i = 0; i < v; i++) {
        list.add(v);
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

빈 배열에 추가, 삭제하기

import java.util.*;

class Solution {
  public int[] solution(int[] arr, boolean[] flag) {
    ArrayList<Integer> list = new ArrayList<>();
    
    for (int i = 0; i < arr.length; i++) {
      if (flag[i]) {
        for (int j = 0; j < arr[i] * 2; j++) {
          list.add(arr[i]);
        }
      } else {
        for (int j = 0; j < arr[i]; j++) {
          list.remove(list.size()-1);
        }
      }
    }
    
    int[] answer = new int[list.size()];
    
    for (int i = 0; i < list.size(); i++) {
      answer[i] = list.get(i);
    }
    
    return answer;
  }
}

배열 만들기 6

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    List<Integer> stkList = new ArrayList<Integer>();
    int[] stk = {-1};
    
    for (int i = 0; i < arr.length; i++) {
      int arr_i = arr[i];
      
      if (stkList.isEmpty()) {
        stkList.add(arr_i);
          
      } else {
        int stk_last = stkList.get(stkList.size() - 1);
        
        if (stk_last == arr_i) {
          stkList.remove(stkList.size() - 1);
        } else {
          stkList.add(arr_i);
        }

      }
    }
    
    if (!stkList.isEmpty()) {
      stk = new int[stkList.size()];
      
      for (int i = 0; i < stkList.size(); i++) {
        stk[i] = stkList.get(i);
      }
    }
    
    return stk;
  }
}

위/아래 코드 시간복잡도는 같지만,
양쪽 끝에서의 삽입/삭제 시 ArrayList보다 ArrayDeque가 더 빠르고 효율적입니다.

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    Deque<Integer> stack = new ArrayDeque<>();
    int[] stk = {-1};
    
    for (int i = 0; i < arr.length; i++) {
      if (stack.isEmpty()) {
        stack.push(arr[i]);
      } else {
        if (stack.peek() == arr[i]) {
          stack.pop();
        } else {
          stack.push(arr[i]);
        }
      }
    }
    
    if (!stack.isEmpty()) {
      stk = new int[stack.size()];
      
      for (int i = stack.size() - 1; i >= 0; i--) {
        stk[i] = stack.pop();
      }
    }
    
    return stk;
  }
}

무작위로 K개의 수 뽑기

import java.util.*;

class Solution {
  public int[] solution(int[] arr, int k) {
      
    int[] answer = new int[k];
    Arrays.fill(answer, -1);
    
    Set<Integer> set = new LinkedHashSet<>();
    
    // 중복 숫자 제거
    for(int i = 0; i < arr.length; i++) {
      set.add(arr[i]);
    }
    
    // k번째 미만에 값 채우기
    int i = 0;
    
    for (int v : set) {
      if (i == k) {
        break;
      }
      
      answer[i] = v;
      
      i++;
    }
    
    return answer;
  }
}

이차원 배열 문제 풀이

특별한 이차원 배열 1

class Solution {
  public int[][] solution(int n) {
    int[][] answer = new int[n][];
    
    for (int i = 0; i < n; i++) {
      answer[i] = new int[n];
      answer[i][i] = 1;
    }
    
    return answer;
  }
}

n x n 배열이므로, 아래처럼 배열 초기화 시 모두 생성해도 됩니다.

class Solution {
  public int[][] solution(int n) {
    int[][] answer = new int[n][n];
    
    for (int i = 0; i < n; i++) {
      answer[i][i] = 1;
    }
    
    return answer;
  }
}

정수를 나선형으로 배치하기

import java.util.*;

class Solution {
  public int[][] solution(int n) {
    int[][] answer = new int[n][n];
    HashMap<String, int[]> moveMap = new HashMap<>();
    
    moveMap.put("up", new int[]{-1, 0});
    moveMap.put("down", new int[]{1, 0});
    moveMap.put("left", new int[]{0, -1});
    moveMap.put("right", new int[]{0, 1});
    
    int curCol = 0;
    int curRow = 0;
    int v = 1;
    answer[0][0] = v;
    String nextMoveKey = "right";
    
    while (v < n * n) {
      int[] nextMove = moveMap.get(nextMoveKey);
      
      int nextRow = curRow + nextMove[0];
      int nextCol = curCol + nextMove[1];
      
      // 배열 밖으로 나가거나, 이미 값이 있는 경우 방향 전환
      if (nextCol >= n || nextRow >= n || nextCol < 0 || nextRow < 0 || answer[nextRow][nextCol] != 0) {
        if (nextMoveKey.equals("right")) {
          nextMoveKey = "down";
        } else if (nextMoveKey.equals("down")) {
          nextMoveKey = "left";
        } else if (nextMoveKey.equals("left")) {
          nextMoveKey = "up";
        } else if (nextMoveKey.equals("up")) {
          nextMoveKey = "right";
        }
        continue;
      }
      
      curRow = nextRow;
      curCol = nextCol;
      v++;
      
      answer[curRow][curCol] = v;
    }
    
    return answer;
  }
}

2차원 배열을 나선형으로 채우는 시뮬레이션 문제입니다.
아래 코드가 더 간결하고 직관적입니다.

class Solution {
  public int[][] solution(int n) {
    int[][] answer = new int[n][n];
    int num = 1;
    int x = 0, y = 0;
    
    // 방향 : 오른쪽, 아래, 왼쪽, 위
    int dx[] = {0, 1, 0, -1};
    int dy[] = {1, 0, -1, 0};
    
    int direction = 0;

    while (num <= n * n) {
      answer[x][y] = num++;

      int nx = x + dx[direction]; 
      int ny = y + dy[direction];

      if (nx < 0 || nx >= n || ny < 0 || ny >= n || answer[nx][ny] != 0) {
        // 범위 밖으로 나갔을 때 방향전환
        direction = (direction + 1) % 4;
        
        nx = x + dx[direction];
        ny = y + dy[direction];
      }
      x = nx;
      y = ny;
    }

    return answer;
  }
}

특별한 이차원 배열 2

class Solution {
  public int solution(int[][] arr) {
    int answer = 1;
    
    for (int i = 0; i < arr.length; i++) {
      for (int j = 0; j < arr.length; j++) {
        if (arr[i][j] != arr[j][i]) {
          answer = 0;
          break;
        }
      }
    }
    
    return answer;
  }
}

정사각형으로 만들기

class Solution {
  public int[][] solution(int[][] arr) {
      
    int maxLangth = Math.max(arr.length, arr[0].length);
    
    int[][] answer = new int[maxLangth][maxLangth];
    
    for (int i = 0; i < answer.length; i++) {
      for (int j = 0; j < answer[i].length; j++) {
        if (i < arr.length && j < arr[i].length) {
          answer[i][j] = arr[i][j];
        }
      }
    }
    
    return answer;
  }
}

행이 열보다 더 많은 경우, 이미 0으로 초기화되어 있으므로 0을 입력하지 않아도 됩니다.

class Solution {
  public int[][] solution(int[][] arr) {
      
    int maxLangth = Math.max(arr.length, arr[0].length);
    
    int[][] answer = new int[maxLangth][maxLangth];
    
    for (int i = 0; i < arr.length; i++) {
      System.arraycopy(arr[i], 0, answer[i], 0, arr[i].length);
    }
    
    return answer;
  }
}

System.arraycopy 함수를 이용하면 더 간결하게 배열 값을 복사할 수 있습니다.
arr[i] 배열의 0번째 인덱스부터 answer[i] 배열의 0번째 인덱스에 arr[i] 배열의 길이만큼 복사합니다.

이차원 배열 대각선 순회하기

class Solution {
  public int solution(int[][] board, int k) {
    int answer = 0;
    
    for (int i = 0; i < board.length; i++) {
      for (int j = 0; j < board[i].length; j++) {
        if (i + j <= k) {
          answer += board[i][j];
        }
      }
    }
    
    return answer;
  }
}

함수 문제 풀이

배열의 길이를 2의 거듭제곱으로 만들기

class Solution {
  public int[] solution(int[] arr) {
    int targetNum = 0;
    int i = 0;
    
    if (arr.length == 1) {
      // 1은 2의 0승이므로 그대로 리턴
      return arr;
    }
        
    while (targetNum < arr.length) {
      i++;
      targetNum = (int) Math.pow(2, i);
    }
    
  int[] answer = new int[targetNum];
  
  for (int j = 0; j < arr.length; j++) {
    answer[j] = arr[j];
  }
  
  return answer;
  }
}

아래처럼, Math.pow 함수 없이 구현할 수도 있습니다.

import java.util.*;

class Solution {
  public int[] solution(int[] arr) {
    int length = 1;
    
    while (length < arr.length) {
      length *= 2;
    }
    
    return Arrays.copyOf(arr, length);
  }
}

배열 비교하기

class Solution {
  public int solution(int[] arr1, int[] arr2) {
    int answer = 0;
    
    if (arr1.length > arr2.length) {
      answer = 1;
    } else if (arr1.length < arr2.length) {
      answer = -1;
    } else {
      int arr1Sum = 0;
      for (int v : arr1) {
        arr1Sum += v;
      }
      
      int arr2Sum = 0;
      for (int v : arr2) {
        arr2Sum += v;
      }
      
      if (arr1Sum > arr2Sum) {
        answer = 1;
      } else if (arr1Sum < arr2Sum) {
        answer = -1;
      } else {
        answer = 0;
      }
    }
    
    return answer;
  }
}

문자열 묶기

import java.util.HashMap;

class Solution {
  public int solution(String[] strArr) {
      
    HashMap<Integer, Integer> countMap = new HashMap<>();
    
    for (String str : strArr) {
      int key = str.length();
      countMap.put(key, countMap.getOrDefault(key, 0) +1);
    }
    
    int answer = 0;
    
    for (int count : countMap.values()) {
      answer = Math.max(answer, count);
    }
    
    return answer;
  }
}

배열의 길이에 따라 다른 연산하기

class Solution {
  public int[] solution(int[] arr, int n) {
    if (arr.length % 2 == 1) {
      for (int i = 0; i < arr.length; i++) {
        if (i % 2 == 0) {
          arr[i] += n;
        }
      }
    } else {
      for (int i = 0; i < arr.length; i++) {
        if (i % 2 == 1) {
          arr[i] += n;
        }
      }
    }
    
    return arr;
  }
}

아래처럼, 하나의 for문으로 구현할 수도 있습니다.

class Solution {
  public int[] solution(int[] arr, int n) {
    for (int i = arr.length % 2 == 1? 0 : 1; i < arr.length; i+=2) {
      arr[i] += n;
    }
    
    return arr;
  }
}

뒤에서 5등까지

import java.util.*;

class Solution {
  public int[] solution(int[] num_list) {
    Arrays.sort(num_list);
    
    return Arrays.copyOfRange(num_list, 0, 5);
  }
}

뒤에서 5등 위로

import java.util.*;

class Solution {
  public int[] solution(int[] num_list) {
    Arrays.sort(num_list);
    
    return Arrays.copyOfRange(num_list, 5, num_list.length);
  }
}

전국 대회 선발 고사

import java.util.*;

class Solution {
  public int solution(int[] rank, boolean[] attendance) {
    int answer = 0;
    
    TreeMap<Integer, Integer> treeMap = new TreeMap<>();
    
    for (int i = 0; i < rank.length; i++) {
      if (attendance[i]) {
        treeMap.put(rank[i], i);
      }
    }
    
    int i = 0;
    
    for (int rankIndex : treeMap.values()) {
      if (i == 3) {
        break;
      }
      
      switch (i) {
        case 0 : 
          answer += (10000 * rankIndex);
          break;
        case 1 : 
          answer += (100 * rankIndex);
          break;
        case 2 : 
          answer += rankIndex;
          break;
      }
      
      i++;
    }
    
    return answer;
  }
}

TreeMap으로 등수에 따라 오름차순 정렬하여 풀이하였습니다.
아래와 같이, 우선순위 큐를 이용하면 값을 순차적으로 꺼낼 수 있어서 간결한 풀이가 가능합니다.

import java.util.*;

class Solution {
  public int solution(int[] rank, boolean[] attendance) {
    PriorityQueue<Integer> pq = new PriorityQueue<>((a, b) -> rank[a] - rank[b]);

    for (int i = 0; i < attendance.length; i++) {
      if (attendance[i]) {
        pq.add(i);
      }
    }

    return 10000 * pq.poll() + 100 * pq.poll() + pq.poll();
  }
}

정수 부분

class Solution {
  public int solution(double flo) {
    return (int) flo;
  }
}

문자열 정수의 합

class Solution {
  public int solution(String num_str) {
    int answer = 0;
    String[] num_arr = num_str.split("");
    
    for (String num : num_arr) {
      answer += Integer.parseInt(num);
    }
    
  return answer;
  }
}

문자열을 정수로 변환하기

class Solution {
  public int solution(String n_str) {
    return Integer.parseInt(n_str);
  }
}

0 떼기

class Solution {
  public String solution(String n_str) {
    String answer = "";
    int length = n_str.length();
    
    for (int i = 0; i < length; i++) {
      char c = n_str.charAt(0);
      
      if (c == '0') {
        n_str = n_str.replaceFirst("0", "");
              
      } else {
          answer = n_str;
        break;
      }
    }
    
    return answer;
  }
}

아래와 같이, 정수로 변환 후 문자열로 변환해도 왼쪽의 0을 제거할 수 있습니다.

class Solution {
  public String solution(String n_str) {
    return String.valueOf(Integer.parseInt(n_str));
  }
}

두 수의 합

import java.math.*;

class Solution {
  public String solution(String a, String b) {
    BigInteger bigA = new BigInteger(a);
    BigInteger bigB = new BigInteger(b);
    
    return bigA.add(bigB).toString();
  }
}

Integer 최대 10자리, Long 최대 19자리 숫자 범위를 초과하는 정수입니다.
math 패키지의 BigInteger 타입을 사용하면 처리할 수 있습니다.

문자열로 변환

class Solution {
  public String solution(int n) {
    return String.valueOf(n);
  }
}

Leave a comment