溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

詳解在Swift中使用KVO

發(fā)布時間:2020-07-21 11:42:02 來源:億速云 閱讀:387 作者:小豬 欄目:開發(fā)技術(shù)

小編這次要給大家分享的是詳解在Swift中使用KVO,文章內(nèi)容豐富,感興趣的小伙伴可以來了解一下,希望大家閱讀完這篇文章之后能夠有所收獲。

在文字的開頭,先說一個小細(xì)節(jié),swift中聲明一個類,你可以集成自NSObject,也可以選擇忽略,二者有什么區(qū)別呢。根據(jù)自己的經(jīng)驗,我得出以下結(jié)論。不足之處,請指出。exmple:我們聲明這樣一個類

class Person: NSObject {
  var name: String?
  override init() {
    super.init()
  }
}
此類打印出的內(nèi)存地址是0x00000fbd00007ffeefbfc240

這段代碼是不會報錯的,是一個典型的swift遺留ObjC語法的寫法,但是如果我們?nèi)サ鬘SObject并打印出他的內(nèi)存地址,如下

class Person {
  var name: String?
  init() {
    
  }
}
此類打印出的內(nèi)存地址是0x00007ffeefbfc240
  • 內(nèi)存地址不一樣,繼承自NSObject的類對象的內(nèi)存地址明顯長度多了8個長度,why?多出的8個空間就是為了存放ObjC對象內(nèi)的isa指針,有興趣的可以往下研究。
  • 繼承自NSObject的類可以使用OC里的一些騷操作,比如KVC、KVO、runtime,否則使用setValue-forKey時是會報錯的。

區(qū)別還有很多,平時在開發(fā)中大家可以多注意這一區(qū)別。個人偏向不繼承NSObject,尤其是我需要此類做一些騷操作時,比如KVO。

KVO是OC一個對象屬性的特性,由于是面向字符串,所以開發(fā)時需要尤其小心,這種奔潰只有執(zhí)行到了才會報錯。聲明如下類:

class Person: NSObject {
   @objc var age: Int?
   var name: String?
   var observation: NSKeyValueObservation?
   
   override init() {
    super.init()
    self.observation = observe(\Person.age, options: .new, changeHandler: { (person, change) in
      print("Person.age的新值 = ", change.newValue as Any)
    })
  }
}

在外部我們,初始化一個對象,并對age進行賦值,如下

let person = Person()
person.age = 18
person.setValue(100, forKey: "age")

程序執(zhí)行后,(ÒωÓױ)!為什么只有一個打?。堪蠢碚f是應(yīng)該打印Person.age的新值 = 18Person.age的新值 = 100的呀,然而并沒有:laughing:。問題出在哪,原來,swift中如果需要對一個值進行監(jiān)聽,那么一定要記住2個關(guān)鍵詞

  • @objc
  • dynamic

否則,

沒有@objc程序在監(jiān)聽時會觸發(fā)奔潰;沒有dynamic則屬性的set方法不會生效,自然就沒有上面的打印,因為KVO的本質(zhì)就是監(jiān)聽屬性的set方法,而可變數(shù)組的增刪操作都不會生效;

但是為什么KVC的操作卻能生效呢?這是因為KVC內(nèi)部的實現(xiàn)過程是

  • [person willChangeValueForKey:@"age"];
  • person->_age = 10;
  • [person didChangeValueForKey:@"age"];
  • 而didChangeValueForKey:內(nèi)部會調(diào)用observe的observeValueForKeyPath:ofObject:change:context:的方法,也就觸發(fā)了KVO

所以正確的寫法應(yīng)該是

class Person: NSObject {
   @objc dynamic var age: Int?
   var name: String?
   var observation: NSKeyValueObservation?
   
   override init() {
    super.init()
    self.observation = observe(\Person.age, options: .new, changeHandler: { (person, change) in
      print("Person.age的新值 = ", change.newValue as Any)
    })
  }
}

看完這篇關(guān)于詳解在Swift中使用KVO的文章,如果覺得文章內(nèi)容寫得不錯的話,可以把它分享出去給更多人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI