[Algorithm] 백준 - 14891번 톱니바퀴 (Swift)

2024. 6. 29. 13:28· → Problems
목차
  1. 문제 소개
  2. 문제 풀이
  3. 톱니바퀴 돌리기
  4. 같이 돌아가는 톱니바퀴 그룹화
  5. 점수 계산
  6. 정답 코드
  7. 디버깅

문제 소개

https://www.acmicpc.net/problem/14891
엄청난 구현문제... (?)
 

문제 풀이

톱니바퀴 돌리기

큐로 구현하는것이 일반적이지만 여기서는 배열의 갯수와 성분들이 작아서 간단하게 removeLast, removeFirst 메서드로 구현했다.

func rotateGear(_ i: Int, _ isClockwise: Bool) {
    if isClockwise {
        gears[i].insert(gears[i].removeLast(), at: gears[i].startIndex)
    } else {
        gears[i].append(gears[i].removeFirst())
    }
}

 

같이 돌아가는 톱니바퀴 그룹화

union, find 알고리즘을 사용했다. 결국 같은 그룹안에 있어야 톱니바퀴가 돌아가는 것이기 때문에 그룹을 짓고 parents 를 반환하는 함수를 만들었다.

func groupGears(_ n: Int) -> [Int] {
    var parents = [0, 1, 2, 3]
    
    func find(_ a: Int) -> Int {
        return parents[a] == a ? a : find(parents[a])
    }
    
    func union(_ a: Int, _ b: Int) {
        let a = find(a)
        let b = find(b)
        parents[b] = a
    }
    
    for i in 0..<3 {
        if gears[i].prefix(3).last != gears[i+1].prefix(7).last {
            union(i, i+1)
        }
    }
    
    return parents
}

 

점수 계산

이 부분은 단순하게 문제에서 주어진대로 구현했다.

func getScore() -> Int {
    var ans = 0
    
    for i in gears.indices {
        if gears[i].first == "1" {
            ans += Int(pow(2.0, Double(i)))
        }
    }
    
    return ans
}

 

정답 코드

import Foundation

var gears = [String]()

for _ in 0..<4 {
    let gear = readLine()!
    gears.append(gear)
}

let k = Int(readLine()!)!

// 같이 돌아가는 톱니바퀴 그룹화
func groupGears(_ n: Int) -> [Int] {
    var parents = [0, 1, 2, 3]
    
    func find(_ a: Int) -> Int {
        return parents[a] == a ? a : find(parents[a])
    }
    
    func union(_ a: Int, _ b: Int) {
        let a = find(a)
        let b = find(b)
        parents[b] = a
    }
    
    for i in 0..<3 {
        if gears[i].prefix(3).last != gears[i+1].prefix(7).last {
            union(i, i+1)
        }
    }
    
    return parents
}

// 톱니바퀴 돌리기
func rotateGear(_ i: Int, _ isClockwise: Bool) {
    if isClockwise {
        gears[i].insert(gears[i].removeLast(), at: gears[i].startIndex)
    } else {
        gears[i].append(gears[i].removeFirst())
    }
}

// 점수 구하기
func getScore() -> Int {
    var ans = 0
    
    for i in gears.indices {
        if gears[i].first == "1" {
            ans += Int(pow(2.0, Double(i)))
        }
    }
    
    return ans
}

for _ in 0..<k {
    let nd = readLine()!.components(separatedBy: " ").map { Int($0)! }
    let n = nd[0] - 1
    let d = nd[1] == 1 ? true : false // 시계방향 true, 반시계방향 false
    
    let parents = groupGears(n)
        
    for i in 0..<4 {
        if parents[n] == parents[i] {
            if abs(n-i) % 2 == 0 {
                rotateGear(i, d)
            } else {
                rotateGear(i, !d)
            }
        }
    }
}

print(getScore())

 

디버깅

아래와 같이 gears 를 입력받으면 모양을 출력해주는 함수를 임의로 구현하여 검증했다.

func printGears(_ gears: [String]) {
    func get(_ i: Int, _ j: Int) -> String {
        return String(gears[i].prefix(j+1).last ?? "?")
    }
    
    print("  \(get(0, 0))     \(get(1, 0))     \(get(2, 0))     \(get(3, 0))  ")
    print(" \(get(0, 7)) \(get(0, 1))   \(get(1, 7)) \(get(1, 1))   \(get(2, 7)) \(get(2, 1))   \(get(3, 7)) \(get(3, 1)) ")
    print("\(get(0, 6))   \(get(0, 2)) \(get(1, 6))   \(get(1, 2)) \(get(2, 6))   \(get(2, 2)) \(get(3, 6))   \(get(3, 2))")
    print(" \(get(0, 5)) \(get(0, 3))   \(get(1, 5)) \(get(1, 3))   \(get(2, 5)) \(get(2, 3))   \(get(3, 5)) \(get(3, 3)) ")
    print("  \(get(0, 4))     \(get(1, 4))     \(get(2, 4))     \(get(3, 4))  ")
}

printGears(gears)

// 출력 값
  1     1     1     0  
 1 1   0 1   0 1   1 0 
1   0 1   1 1   0 0   0
 1 1   0 1   1 0   0 0 
  0     1     1     0

'→ Problems' 카테고리의 다른 글

[Algorithm] 프로그래머스 - 도넛과 막대 그래프 (Swift)  (0) 2024.07.05
[Algorithm] 백준 - 11723번 집합 (Swift)  (0) 2024.07.02
[Algorithm] 백준 - 4375번 1 (Swift)  (0) 2024.06.27
[Algorithm] 백준 - 14499번 주사위 굴리기 (Swift)  (0) 2024.06.25
[Algorithm] 백준 - 2252번 줄세우기 (Swift) (위상정렬)  (0) 2024.06.22
  1. 문제 소개
  2. 문제 풀이
  3. 톱니바퀴 돌리기
  4. 같이 돌아가는 톱니바퀴 그룹화
  5. 점수 계산
  6. 정답 코드
  7. 디버깅
'→ Problems' 카테고리의 다른 글
  • [Algorithm] 프로그래머스 - 도넛과 막대 그래프 (Swift)
  • [Algorithm] 백준 - 11723번 집합 (Swift)
  • [Algorithm] 백준 - 4375번 1 (Swift)
  • [Algorithm] 백준 - 14499번 주사위 굴리기 (Swift)
Swift librarian
Swift librarian
Swift librarian
Swift Library
Swift librarian
전체
오늘
어제
  • 분류 전체보기 (231)
    • 📺 Programming (5)
    • → Architecture (2)
    • → Design Pattern (0)
    • → Computer Science (15)
    • ⚙️ Algorithm (0)
    • → 알고리즘 관련 (22)
    • → Problems (104)
    • 🚀 Project (0)
    • → 알쏭달쏭 (0)
    • → Shook (2)
    • → Solver (8)
    • → Taster (7)
    • → Outline (4)
    • → Pointer (2)
    • → Guesser (3)
    • 🦜 Swift (2)
    • → Swift Archive (12)
    • → Swift Study (12)
    • → Xcode (6)
    • 🧰 Framework (0)
    • → Foundation (1)
    • → UIKit (2)
    • → SwiftUI (3)
    • → CoreData (2)
    • → MapKit (1)
    • → CoreHaptic (1)
    • → User Notification (1)
    • → StoreKit (2)
    • 🏛️ Library (0)
    • → TCA (0)
    • 🐈‍⬛ Git (8)
    • → Git의 원리 (2)
    • → Git 심화 (1)
    • 📦 Other (1)
    • 👦🏻 Log (0)

최근 글

hELLO · Designed By 정상우.v4.2.2
Swift librarian
[Algorithm] 백준 - 14891번 톱니바퀴 (Swift)
상단으로

티스토리툴바

단축키

내 블로그

내 블로그 - 관리자 홈 전환
Q
Q
새 글 쓰기
W
W

블로그 게시글

글 수정 (권한 있는 경우)
E
E
댓글 영역으로 이동
C
C

모든 영역

이 페이지의 URL 복사
S
S
맨 위로 이동
T
T
티스토리 홈 이동
H
H
단축키 안내
Shift + /
⇧ + /

* 단축키는 한글/영문 대소문자로 이용 가능하며, 티스토리 기본 도메인에서만 동작합니다.