[swift] Swift DeepDiff를 사용하여 컬렉션 데이터 변경 시 해당하는 셀만 업데이트하기

개요

iOS 앱을 개발하다 보면 테이블 뷰나 컬렉션 뷰 등에서 동적으로 데이터를 업데이트해야 할 때가 있습니다. 이때, 변화가 있는 데이터만 업데이트하고, 그 외에는 그대로 유지할 수 있는 방법이 필요합니다. Swift DeepDiff 라이브러리는 컬렉션 데이터의 변경점을 식별할 수 있는 기능을 제공하여 이러한 문제를 해결할 수 있습니다. 이번 글에서는 Swift DeepDiff를 사용하여 컬렉션 데이터 변경 시 해당하는 셀만 업데이트하는 방법에 대해 알아보겠습니다.

DeepDiff 설치하기

먼저, 프로젝트에 Swift DeepDiff를 설치해야 합니다. DeepDiff는 CocoaPods나 Swift Package Manager를 통해 설치할 수 있습니다.

CocoaPods를 사용하는 경우

pod 'DeepDiff'

Swift Package Manager를 사용하는 경우

Xcode에서 프로젝트를 열고, File -> Swift Packages -> Add Package Dependency를 선택합니다. 다음 URL을 입력하고, 버전을 선택한 다음, Add Package를 클릭합니다.

https://github.com/onmyway133/DeepDiff.git

DeepDiff를 설치한 후에는 import DeepDiff를 코드 상단에 추가하여 사용할 수 있습니다.

사용법

DeepDiff를 사용하여 컬렉션 데이터 변경 시 해당하는 셀만 업데이트하는 방법은 다음과 같습니다.

  1. 이전 데이터와 새로운 데이터를 비교하여 변경점을 식별합니다. ```swift let oldData = [1, 2, 3, 4, 5] let newData = [1, 2, 3, 6, 7]

let changeset = diff(old: oldData, new: newData)


2. 변경점을 기반으로 적용할 업데이트를 식별합니다.
```swift
let updateItems = changeset.updatedItems.compactMap { $0.item }
  1. 이전 데이터와 현재 데이터를 업데이트합니다.
    oldData = newData
    
  2. 업데이트된 데이터만 화면에 반영합니다.
    tableView.beginUpdates()
    for item in updateItems {
     let indexPath = IndexPath(row: item, section: 0)
     tableView.reloadRows(at: [indexPath], with: .none)
    }
    tableView.endUpdates()
    

위 코드에서 변경점을 식별하기 위해 diff(old:new:) 함수를 사용했습니다. 변경된 항목을 식별하기 위해서는 changeset.updatedItems를 사용하면 됩니다. 업데이트된 데이터를 화면에 반영하기 위해서는 tableView.reloadRows(at:with:) 메서드를 사용하면 됩니다.

예제

다음은 DeepDiff를 사용하여 컬렉션 데이터 변경 시 해당하는 셀만 업데이트하는 예제입니다.

import UIKit
import DeepDiff

class ViewController: UIViewController {
    
    @IBOutlet weak var tableView: UITableView!
    
    var data = [1, 2, 3, 4, 5]
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        tableView.dataSource = self
        tableView.delegate = self
    }
    
    @IBAction func updateButtonTapped(_ sender: UIButton) {
        let newData = [1, 2, 3, 6, 7]
        
        let changeset = diff(old: data, new: newData)
        let updateItems = changeset.updatedItems.compactMap { $0.item }
        
        data = newData
        
        tableView.beginUpdates()
        for item in updateItems {
            let indexPath = IndexPath(row: item, section: 0)
            tableView.reloadRows(at: [indexPath], with: .none)
        }
        tableView.endUpdates()
    }
}

extension ViewController: UITableViewDataSource, UITableViewDelegate {
    
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return data.count
    }
    
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath)
        cell.textLabel?.text = "\(data[indexPath.row])"
        return cell
    }
    
}

위 예제에서 updateButtonTapped(_:) 메서드는 버튼을 탭했을 때 데이터를 업데이트하고, 이전 데이터와 현재 데이터를 비교하여 변경된 셀만 업데이트합니다. tableView(_:cellForRowAt:) 메서드는 테이블 뷰의 각 셀에 데이터를 표시합니다.

결론

Swift DeepDiff는 컬렉션 데이터 변경 시 해당하는 셀만 업데이트하는 데 유용한 기능을 제공합니다. 변경점을 식별하고, 업데이트 된 데이터를 화면에 반영하는 과정을 단순하게 해주므로 개발 과정에서 효율적으로 사용할 수 있습니다.