deinit到底在做什么?为什么?

时间:2017-05-20 17:55:04

标签: ios swift firebase firebase-realtime-database deinit

我的问题与为Firebase提供的示例代码有关。您可以在GithubDocumentation上找到它。无论如何,我真的不明白" deinit"关于数据库引用正在做什么。我阅读了deinit swift文档here。我想我理解使用deinit的目的,但我不确定它在这种情况下是如何工作的。

let kBannerAdUnitID = "ca-app-pub-3940256099942544/2934735716"

@objc(FCViewController)
class FCViewController: UIViewController, UITableViewDataSource, UITableViewDelegate,
    UITextFieldDelegate, UIImagePickerControllerDelegate, UINavigationControllerDelegate,
        InviteDelegate {

  // Instance variables
  @IBOutlet weak var textField: UITextField!
  @IBOutlet weak var sendButton: UIButton!
  var ref: DatabaseReference!
  var messages: [DataSnapshot]! = []
  let posts = [Post]()

  var msglength: NSNumber = 10
  fileprivate var _refHandle: DatabaseHandle!

  var storageRef: StorageReference!
  var remoteConfig: RemoteConfig!

  @IBOutlet weak var banner: GADBannerView!
  @IBOutlet weak var clientTable: UITableView!

  override func viewDidLoad() {
    super.viewDidLoad()

    self.clientTable.register(UITableViewCell.self, forCellReuseIdentifier: "tableViewCell")

    configureDatabase()
    configureStorage()
    configureRemoteConfig()
    fetchConfig()
    loadAd()
    logViewLoaded()
  }

    deinit {

        // NOT REALLY SURE WHAT THIS DOES
        if let refHandle = _refHandle {
            self.ref.child("messages").removeObserver(withHandle: _refHandle)
        }
    }

func configureDatabase() {
    ref = Database.database().reference()
    // Listen for new messages in the Firebase database
    _refHandle = self.ref.child("messages").observe(.childAdded, with: { [weak self] (snapshot) -> Void in
        guard let strongSelf = self else { return }

        // EACH SNAPSHOT IS PLACED IN THE MESSAGES ARRAY
        strongSelf.messages.append(snapshot)

        // MAKE THE NUMBER OF ROWS IN THE FEED == THE NUMBER OF SNAPSHOTS
        strongSelf.clientTable.insertRows(at: [IndexPath(row: strongSelf.messages.count-1, section: 0)], with: .automatic)
    })
}

1 个答案:

答案 0 :(得分:1)

在初始化期间,视图控制器会将childAdded观察者添加到给定的Firebase数据库节点。在视图控制器早已消失(即,解除分配)之后,传入的闭包仍将被称为甚至。 Firebase保留了引用(顺便提一下,对视图控制器的引用)。这种不必要的呼叫可能不是一个主要问题(但是!),但它仍然是错误的。

deinit代码可防止这种情况发生。

在取消分配视图控制器之前立即调用 deinitializer 。因此,它为上述数据库节点中的删除观察者提供了一个不错的位置。观察者(引用)存储在_refHandle属性中。

顺便说一句,你可以简化你的观察者删除代码:

if let refHandle = _refHandle {
    self.ref.removeObserver(withHandle: _refHandle)
}

您无需在注册的相同节点上调用removeObserver

弱视图控制器。上述闭包中的[weak self]限定符指定Firebase应仅保留对视图控制器的弱引用。这对于确保视图控制器在不再需要时从内存中正确释放至关重要。没有它,你的deinit永远不会被召唤!