1. 程式人生 > 程式設計 >在Swift中使用KVO的細節以及內部實現解析(推薦)

在Swift中使用KVO的細節以及內部實現解析(推薦)

在文字的開頭,先說一個小細節,swift中宣告一個類,你可以整合自NSObject,也可以選擇忽略,二者有什麼區別呢。根據自己的經驗,我得出以下結論。不足之處,請指出。exmple:我們宣告這樣一個類

class Person: NSObject {
  var name: String?
  override init() {
    super.init()
  }
}
此類打印出的記憶體地址是0x00000fbd00007ffeefbfc240

這段程式碼是不會報錯的,是一個典型的swift遺留ObjC語法的寫法,但是如果我們去掉NSObject並打印出他的記憶體地址,如下

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

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

KVO是OC一個物件屬性的特性,由於是面向字串,所以開發時需要尤其小心,這種奔潰只有執行到了才會報錯。宣告如下類:

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")

程式執行後,(ÒωÓױ)!為什麼只有一個列印?按理說是應該列印Person.age的新值 = 18Person.age的新值 = 100的呀,然而並沒有:laughing:。問題出在哪,原來,swift中如果需要對一個值進行監聽,那麼一定要記住2個關鍵詞

  • @objc
  • dynamic

否則,

沒有@objc程式在監聽時會觸發奔潰;沒有dynamic則屬性的set方法不會生效,自然就沒有上面的列印,因為KVO的本質就是監聽屬性的set方法,而可變陣列的增刪操作都不會生效;

但是為什麼KVC的操作卻能生效呢?這是因為KVC內部的實現過程是

  • [person willChangeValueForKey:@"age"];
  • person->_age = 10;
  • [person didChangeValueForKey:@"age"];
  • 而didChangeValueForKey:內部會呼叫observe的observeValueForKeyPath:ofObject:change:context:的方法,也就觸發了KVO

所以正確的寫法應該是

class Person: NSObject {
   @objc dynamic var age: Int?
   var name: String?
   var observation: NSKeyValueObservation?
   
   override init() {
    super.init()
    self.observation = observe(\Person.age,change.newValue as Any)
    })
  }
}

到此這篇關於在Swift中使用KVO的細節以及內部實現解析的文章就介紹到這了,更多相關Swift使用KVO內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!