How use closure in your project?
This post hasn't been updated for 6 years
Part 1: https://viblo.asia/p/gioi-thieu-ve-closure-trong-swift-ios-1Je5E8z0lnL
Hello mọi người.
Do có bạn comment hỏi thêm về closure ở bài Part 1 nên mình mạn phép viết thêm 1 chút nữa về closure.
Uỷ thác sự kiện trong Uitableview
Bạn có một sự kiện nhấn vào button ở trong 1 cell trong 1 uitableViewController để gọi một API show detail video chẳng hạn sao đó sẽ redirect đến màn videoDetailViewController.
# Trong cell bạn đặt một closure để uỷ thác cho uitableView
# VideoModel là model chứa thông tin của video trong 1 cell
class ExampleCell {
var didSelectDetailVideoButton: ((VideoModel?) -> Void)?
//Mark: Sự kiện action click của button trong cell
@IBAction private func didSelectDetailButton(_ sender: UIButton) {
didSelectDetailVideoButton?(video)
}
}
// Trong uitableView. Trong datasource của UITableViewDataSource
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
guard let cell = tableView.dequeueReusableCell(withIdentifier: "CELL", for: indexPath) as? ExampleCell else {
return UITableViewCell()
}
cell.didSelectDetailVideoButton = { [weak self] video in
// TODO: Giờ bạn có thể thực hiện code chuyển màn hình ở đây với video được truyền ra bởi cell rồi.
}
return cell
}
Chú ý: Bạn có để ý Chỗ closure mình để weak self không. Là bởi vì khi thực hiện closure thì cell và uitableview sẽ có một liên kết strong để có thể truyền tải dữ liệu callback. Nếu bạn k để weak self ở đây thì sẽ có thể xảy ra retain cycles dẫn đến uitablview or cell không được giải phóng.
cell.didSelectDetailVideoButton = { [weak self] video in
// TODO: Giờ bạn có thể thực hiện code chuyển màn hình ở đây với video được truyền ra bởi cell rồi.
}
Ngoài ra bạn có thể áp dụng với các trường hợp với collectview hay với viewDetail hay một view popup uỷ thác cho view cha của nó thực hiện 1 tác vụ nào đó. Tuy nhiên có một cách khác là bạn sử dụng Protocol để có thể thực hiện tác vụ y hệt như trên.
Sử dụng Protocol
Trong cell. Ta khai báo protocol là SendBackDataProtocol
Trong cell có 1 button. Khi ấn vào button đó ta sẽ truyên dữ liệu vào func sendBackDataFunc
import UIKit
protocol SendBackDataProtocol: class {
func sendBackDataFunc(data: String?)
}
class testProtocolCell: UITableViewCell {
var delegate: SendBackDataProtocol?
@IBAction func handleSendBackData(_ sender: Any) {
self.delegate?.sendBackDataFunc(data: "Send back data")
}
}
Trong uitableView
import UIKit
import AVFoundation
class ViewController: UIViewController, UITableViewDelegate, UITableViewDataSource, SendBackDataProtocol {
@IBOutlet weak var uitableView: UITableView!
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return 1
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: "testProtocolCell") as! testProtocolCell
cell.delegate = self
return cell
}
// Mark: Thực hiện protocol func
func sendBackDataFunc(data: String?) {
print("SEnd back data: \(data)")
}
override func viewDidLoad() {
super.viewDidLoad()
uitableView.delegate = self
uitableView.dataSource = self
uitableView.register(UINib(nibName: "testProtocolCell", bundle: nil), forCellReuseIdentifier: "testProtocolCell")
}
}
Các bạn để ý. Chúng ta implement SendBackDataProtocol ở bên trên class. Thực hiện func protocol sendBackDataFunc in ra print("SEnd back data: (data)") Trong cellForRowAt chúng ta thực hiện
cell.delegate = self
Để có thể nhận delegate trong class uitableView
Kêt quả: Khi bấm 2 lần button
SEnd back data: Optional("Send back data")
SEnd back data: Optional("Send back data")
Uỷ thác khi bạn thực hiện tác vụ gọi API hoặc một tác vụ dưới thread background
Các bạn sử dụng Alamofire hoặc tự code network gọi API của apple. Ở đây mình sử dụng Alamofire để ví dụ.
Ví dụ mình thực hiện 1 action đăng kí tài khoản.
Định nghĩa URL sử dụng Alamofire advance
Các bạn có thể đọc bài của mình về Rxswift Build base Network using Rxswift hoặc các bạn đọc trên Document Alamofire advance using để hiểu câu lệnh dưới.
func register(with email: String, password: String, success: (() -> Void)?, fail: ((String?) -> Void)?) {
let parameters: Parameters = [
"email": email,
"password": password
]
Alamofire.request("https://localhost:3000/sign_up", method: .post, parameters: parameters).responseJSON { response in
switch response.result {
case .success(let value):
// TODO: Bạn xử lý register thành công ở đây như setAccessToken cho những request sau đó.
success?()
case .failure(let error):
// TODO: Nếu có lỗi trả về 1 closure báo lỗi để xử lý.
fail?("Register Fail")
}
}
}
Như các bạn thấy. Ở đây mình xử dụng 2 closure chờ kết quả là success và fail để đảm nhiểm truyền trả dữ liệu tuỳ theo trường hợp trả về của API để xử lý.
Goi api ở RegisterViewController
RequestManager.shared là một struct singletons Network chưa các func gọi API. register func chỉ là 1 trong số rất nhiều func.
RequestManager.shared.register(with: registerData, success: { [weak self] in
print("Register success or Redirect to Main screen ")
}) { [weak self] (err) in
print("Has error: \(err)")
}
Và ở trong viewController chúng ta sẽ sử lý dữ liệu được báo về thông qua closure một cách đơn giản như vậy.
Sử dụng closure để action trong UIAlertView
Sử dụng alert thì các bạn sử dụng rất phổ biến trong các trường hợp thông báo cho người dùng. Nhưng đôi sử dụng lại nhiều lần thì ta nên viết thành 1 extension dùng chung để đỡ phải viết đi viết lại nhiều lần
extension UIViewController {
func showAlert(_ title: String, _ message: String, _ okTitle: String, _ okAction: (() -> Void)?, _ cancelTitle: String?, _ cancelAction: (() -> Void)? = nil) {
let alertVC = UIAlertController(title: title, message: message, preferredStyle: .alert)
let okAction = UIAlertAction(title: okTitle, style: .default) { _ in
okAction?()
}
alertVC.addAction(okAction)
if cancelTitle != nil {
let cancelAction = UIAlertAction(title: cancelTitle!, style: .default) { _ in
cancelAction?()
}
alertVC.addAction(cancelAction)
}
self.present(alertVC, animated: true, completion: nil)
}
}
Ở đây mình implement 2 closure là okAction và cancelAction để xử lý trường hợp người dùng bâm ok và bấm cancel trên alert
// Trong viewController chúng ta có.
self.showAlert("Title of alert", "Content of alert", "Ok", { [weak self] in
print("User click ok")
}, "Cancel") { [weak self] in
print("User click cancel")
}
Sử dụng closure pass data giữa các controller.
Các bạn có thể tham khảo các pass data giữa các màn hình ở bài viết của Anh Tình (Master ios) Pass data between viewController
Kết luận
Trên đây thường là các trường hợp bạn sử dụng closure mà mình biết. Ví dụ còn sơ sài do nếu viết hoàn chỉnh thì nó thuộc phần build code base nên sẽ rất dài nên mình như đưa ra các ví dụ cụ thể. Nếu chưa hiểu hay còn gì sai xót bạn có thể comment bên dưới.
Theo mình đánh giá các ưu nhược điểm của closure như sau:
Ưu điểm:
- Ngắn gọn, dễ sử dụng.
- Làm code clear hơn hơn về mặt syntax.
- Xử lý các tình huống dữ liệu bất đồng bộ như gọi API hay xử lý background job.
- Code ít hơn so với sử dụng protocol.
Nhược điểm:
- Một nhược điểm là khó debug và maintain đối với code non tay như mình chẳng hạn. Đó là mình thấy thế không biết các bạn sao.
- Có thể xảy ra retain cycles nếu bạn không cẩn thận.
Tuy nhiên closure rất hữu ích trong các project và đặc biết gần như không thể thiếu (Gần như thối nhé ^^!) Trên đây nhận đinh ngắn của mình. Mình cũng chưa sử dụng quá nhiều closure nên cũng chưa thể hiểu được nó. Tuy nhiên mong các bạn có thể ít ra là sử dụng được closure trong project của mình.
Có gì thắc mắc các bạn cứ comment ở dưới nhé. Thanks for watching.
All Rights Reserved