`
啸笑天
  • 浏览: 3461096 次
  • 性别: Icon_minigender_1
  • 来自: China
社区版块
存档分类
最新评论

KVO原理及知识点(swift2.0)

 
阅读更多

KVO (Key-Value Observing) 是 Cocoa 中公认的最强大的特性之一,但是同时它也以烂到家的 API 和极其难用著称。和属性观察不同,KVO 的目的并不是为当前类的属性提供一个钩子方法,而是为了其他不同实例对当前的某个属性 (严格来说是 keypath) 进行监听时使用的。其他实例可以充当一个订阅者的角色,当被监听的属性发生变化时,订阅者将得到通知。

 

在 Swift 中我们也是可以使用 KVO 的,观察者和被观察者都必须是 NSObject 的子类。这是可以理解的,因为 KVO 是基于 KVC (Key-Value Coding) 以及动态派发技术实现的,而这些东西都是 Objective-C 运行时的概念,这也意味着 Swift 中强大的 Struct,Enum以及泛型都与 KVO 无缘了。另外由于 Swift 为了效率,默认禁用了动态派发,因此想用 Swift 来实现 KVO,我们还需要做额外的工作,那就是将想要观测的对象标记为 dynamic,表示该属性的存取都由 runtime 在运行时来决定。除此之外,在 NSObject 子类中几乎没有属性默认是使用@dynamic 修饰(该关键字最常见场景是在 Core Data 里, NSManagedObject 子类的属性都是 dynamic 的),所以若想对某个属性进行观察,还必须在当前的子类中 override 该属性,override 时,采用 super 的实现即可。

所以Swift要使用KVO必须满足:

  1. 观察者和被观察者都必须是 NSObject 的子类
  2. 监听属性必须标记dynamic

 

原理:

  • 当某个类的对象第一次被观察时,系统就会在运行期动态地创建该类的一个派生类,在这个派生类中重写基类中任何被观察属性的 setter 方法。
  • 派生类在被重写的 setter 方法实现真正的通知机制,键值观察通知依赖于 NSObject 的两个方法: willChangeValueForKey: 和 didChangevlueForKey: 。在一个被观察属性发生改变之前, willChangeValueForKey: 一定会被调用,这就 会记录旧的值。而当改变发生后, didChangeValueForKey: 会被调用,继而 observeValueForKey:ofObject:change:context: 也会被调用。可以手动实现这些调用,但很少有人这么做。这么做是基于设置属性会调用 setter 方法,而通过重写就获得了 KVO 需要的通知机制。当然前提是要通过遵循 KVO 的属性设置方式来变更属性值,如果仅是直接修改属性对应的成员变量,是无法实现 KVO 的。
  • 同时派生类还重写了 class 方法以“欺骗”外部调用者它就是起初的那个类。
  • 系统将这个对象的 isa 指针指向这个新诞生的派生类,因此这个对象就成为该派生类的对象了,因而在该对象上对 setter 的调用就会调用重写的 setter,从而激活键值通知机制。
  • 派生类还重写了 dealloc 方法来释放资源。

看下面代码:

-----------------start------------

对象变量名字: people

对象: <TestKVOKVC.People: 0x7fa449c14a60>

类: TestKVOKVC.People

元类: TestKVOKVC.People

实现的方法: setAge:, setSex:, age, sex, initWithName:age:sex:address:, name, address, .cxx_destruct, init, setName:, setAddress:, 

-----------------end------------

-----------------start------------

对象变量名字: namePeople

对象: <TestKVOKVC.People: 0x7fa449c61e20>

类: TestKVOKVC.People

元类: NSKVONotifying_TestKVOKVC.People

实现的方法: setAge:, setName:, class, dealloc, _isKVOA, 

-----------------end------------

-----------------start------------

对象变量名字: nameAgePeople

对象: <TestKVOKVC.People: 0x7fa449cbe990>

类: TestKVOKVC.People

元类: NSKVONotifying_TestKVOKVC.People

实现的方法: setAge:, setName:, class, dealloc, _isKVOA, 

-----------------end------------

0x00000001083a3c60

0x00000001089807fa

0x00000001083a3c60

0x00000001089807fa

可以看到输出类名始终为:TestKVOKVC.People,这是因为新诞生的派生类重写了 -class 方法声称它就是起初的基类,只有使用 runtime 函数 object_getClass 才能一睹芳容:NSKVONotifying_TestKVOKVC.People。注意看:name,age 两个被观察对象真正的类型都是 NSKVONotifying_TestKVOKVC.People,而且该类实现了:setAge:, setName:, class, dealloc, _isKVOA 这些方法。其中 setAge:, setName:重写实现通知机制, class声明原来的类 , dealloc释放资源,私有方法 _isKVOA 是用来标示该类是一个 KVO 机制声称的类。在这里 swift 做了一些优化,它对所有被观察对象只生成一个派生类,该派生类实现所有被观察对象的 setter 方法,这样就减少了派生类的数量,提供了效率。所有 NSKVONotifying_TestKVOKVC.People 这个派生类重写了 setAge,setName方法(留意:没重写其他set方法)。

接着来看最后两行输出,地址 0x00000001083a3c60 是 TestKVOKVC.People 类中的实现,而地址是 0x00000001089807fa 是派生类 NSKVONotifying_TestKVOKVC.People 类中的实现。那后面那个地址到底是什么呢?可以通过 lldb 的 info 命令 image lookup --address  0x00000001089807fa 查看该地址的信息:

(lldb) image lookup --address  0x00000001089807fa

      Address: Foundation[0x00000000000637fa] (Foundation.__TEXT.__text + 401690)

      Summary: Foundation`_NSSetObjectValueAndNotify

 

看起来它是 Foundation 框架提供的私有函数:_NSSetObjectValueAndNotify。更进一步,我们来看看 Foundation 到底提供了哪些用于 KVO 的辅助函数。打开 terminal,使用 nm -a 命令查看 Foundation 中的信息:

nm -a /System/Library/Frameworks/Foundation.framework/Versions/C/Foundation

其中查找到我们关注的函数:

00000000000c76bf t __NSSetBoolValueAndNotify

0000000000152ab2 t __NSSetBoolValueForKeyInIvar

00000000000c74c8 t __NSSetBoolValueForKeyWithMethod

00000000000997db t __NSSetCharValueAndNotify

0000000000064759 t __NSSetCharValueForKeyInIvar

00000000000646f1 t __NSSetCharValueForKeyWithMethod

000000000001bd73 t __NSSetCheckSize

00000000003f7a50 b __NSSetClass

00000000001019af t __NSSetDirectory2

00000000000ab4d3 t __NSSetDoubleValueAndNotify

0000000000152b5d t __NSSetDoubleValueForKeyInIvar

00000000000abf65 t __NSSetDoubleValueForKeyWithMethod

0000000000139763 T __NSSetExceptionRaiser

0000000000180650 t __NSSetFileModificationUNIXTime

00000000000c9cca t __NSSetFloatValueAndNotify

0000000000152bb6 t __NSSetFloatValueForKeyInIvar

0000000000152712 t __NSSetFloatValueForKeyWithMethod

00000000000c9ba9 t __NSSetIntValueAndNotify

0000000000152c0f t __NSSetIntValueForKeyInIvar

00000000000abefb t __NSSetIntValueForKeyWithMethod

00000000001817a7 T __NSSetLogCStringFunction

00000000000a5389 t __NSSetLongLongValueAndNotify

0000000000152d5f t __NSSetLongLongValueForKeyInIvar

00000000000a55e1 t __NSSetLongLongValueForKeyWithMethod

000000000015abd7 t __NSSetLongValueAndNotify

0000000000152cb7 t __NSSetLongValueForKeyInIvar

0000000000152779 t __NSSetLongValueForKeyWithMethod

000000000011651f T __NSSetMainBundle

00000000001531ad t __NSSetNilValueForKey

00000000000647b0 t __NSSetObjectSetIvarValueForKeyInIvar

0000000000096e38 t __NSSetObjectValueAndNotify

0000000000028b2d t __NSSetObjectValueForKeyInIvar

00000000000c0657 t __NSSetPointValueAndNotify

0000000000152e5f t __NSSetPointValueForKeyInIvar

00000000000c05f1 t __NSSetPointValueForKeyWithMethod

000000000015b067 t __NSSetRangeValueAndNotify

0000000000152ebc t __NSSetRangeValueForKeyInIvar

00000000001528b2 t __NSSetRangeValueForKeyWithMethod

0000000000099610 t __NSSetRectValueAndNotify

0000000000152f15 t __NSSetRectValueForKeyInIvar

000000000015291d t __NSSetRectValueForKeyWithMethod

000000000015ae1f t __NSSetShortValueAndNotify

0000000000152db3 t __NSSetShortValueForKeyInIvar

0000000000152849 t __NSSetShortValueForKeyWithMethod

00000000000af692 t __NSSetSizeValueAndNotify

0000000000152f86 t __NSSetSizeValueForKeyInIvar

00000000000af62d t __NSSetSizeValueForKeyWithMethod

000000000015aab3 t __NSSetUnsignedCharValueAndNotify

0000000000152b09 t __NSSetUnsignedCharValueForKeyInIvar

00000000001526a9 t __NSSetUnsignedCharValueForKeyWithMethod

00000000000e59a2 t __NSSetUnsignedIntValueAndNotify

0000000000152c63 t __NSSetUnsignedIntValueForKeyInIvar

0000000000103f5e t __NSSetUnsignedIntValueForKeyWithMethod

0000000000096fa8 t __NSSetUnsignedLongLongValueAndNotify

00000000000f5f93 t __NSSetUnsignedLongLongValueForKeyInIvar

00000000000c745d t __NSSetUnsignedLongLongValueForKeyWithMethod

000000000015acfa t __NSSetUnsignedLongValueAndNotify

0000000000152d0b t __NSSetUnsignedLongValueForKeyInIvar

00000000001527e1 t __NSSetUnsignedLongValueForKeyWithMethod

000000000015af43 t __NSSetUnsignedShortValueAndNotify

0000000000152e09 t __NSSetUnsignedShortValueForKeyInIvar

0000000000103ef5 t __NSSetUnsignedShortValueForKeyWithMethod

 

Foundation 提供了大部分基础数据类型的辅助函数,此外还包括一些常见的 Cocoa 结构体如 Point, Range, Rect, Size,这表明这些结构体也可以用于自动键值观察,但要注意除此之外的结构体就不能用于自动键值观察了。

//
//  People.swift
//  TestKVOKVC
//
//  Created by yangjun zhu on 15/9/12.
//  Copyright © 2015年 Cactus. All rights reserved.
//

import Foundation


class People: NSObject{
    dynamic var name: String
    dynamic var age: Int
    dynamic var sex: Int
    var address: Address?
    
    init(name: String, age: Int, sex: Int, address: Address){
        self.name = name
        self.age = age
        self.sex = sex
        self.address = address
        
    }
}

 

 

//
//  PrincipleViewController.swift
//  TestKVOKVC
//
//  Created by yangjun zhu on 15/9/12.
//  Copyright © 2015年 Cactus. All rights reserved.
//

import UIKit

class PrincipleViewController: UIViewController {
    let people = People(name: "Owen", age: 1, sex: 1, address: Address())
    let namePeople = People(name: "Owen", age: 1, sex: 1, address: Address())
    let nameAgePeople = People(name: "Owen", age: 1, sex: 1, address: Address())
    
    deinit{
        self.namePeople .removeObserver(self, forKeyPath: "name")
        self.nameAgePeople .removeObserver(self, forKeyPath: "name")
        self.nameAgePeople .removeObserver(self, forKeyPath: "age")
        
        
    }
    
    private var myContext = 0
    
    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.
        
        self.namePeople.addObserver( self, forKeyPath: "name", options: NSKeyValueObservingOptions([.New, .Old]), context: &myContext)
        self.nameAgePeople.addObserver( self, forKeyPath: "name", options: NSKeyValueObservingOptions([.New, .Old]), context: &myContext)
        self.nameAgePeople.addObserver( self, forKeyPath: "age", options: NSKeyValueObservingOptions([.New, .Old]), context: &myContext)
        self.printDescription("people", obj: people)
        self.printDescription("namePeople",obj: namePeople)
        self.printDescription("nameAgePeople",obj: nameAgePeople)
    }
    
    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
    
    
    
    
    
    private func classMethodNames(c: AnyObject) -> [String]
    {
        var arr = [String]()
        var methodCount: CUnsignedInt = 0;
        let methodList = class_copyMethodList(object_getClass(c), &methodCount);
        
        for i in 0...methodCount {
            arr.append( NSStringFromSelector(method_getName(methodList[Int(i)])))
        }
        
        free(methodList);
        
        return arr;
    }
    
    private func printDescription(objectName: String, obj: AnyObject)
    {
        print("-----------------start------------")
        print("对象变量名字:",objectName)
        print("对象:",obj)
        print("类:",NSStringFromClass(obj.classForCoder))
        print("元类:",NSStringFromClass(object_getClass(obj)))
        print("实现的方法:",self.classMethodNames(obj).joinWithSeparator(", "))
        print("-----------------end------------")
        
    }
    
    /*
    func observeValueForKeyPath(keyPath: String?,  object: AnyObject?, change: [String : AnyObject]?, context: UnsafeMutablePointer<Void>) {
    
    if let change = change where context == &myContext{
    print(keyPath, "改变了")
    print(keyPath, "new:" , change[NSKeyValueChangeNewKey])
    print(keyPath, "old:" , change[NSKeyValueChangeOldKey])
    return;
    }
    super.observeValueForKeyPath(keyPath, ofObject: object, change: change, context: context)
    }
    */
    
}

 

输出:

-----------------start------------

对象变量名字: people

对象: <TestKVOKVC.People: 0x7fa449c14a60>

类: TestKVOKVC.People

元类: TestKVOKVC.People

实现的方法: setAge:, setSex:, age, sex, initWithName:age:sex:address:, name, address, .cxx_destruct, init, setName:, setAddress:, 

-----------------end------------

-----------------start------------

对象变量名字: namePeople

对象: <TestKVOKVC.People: 0x7fa449c61e20>

类: TestKVOKVC.People

元类: NSKVONotifying_TestKVOKVC.People

实现的方法: setAge:, setName:, class, dealloc, _isKVOA, 

-----------------end------------

-----------------start------------

对象变量名字: nameAgePeople

对象: <TestKVOKVC.People: 0x7fa449cbe990>

类: TestKVOKVC.People

元类: NSKVONotifying_TestKVOKVC.People

实现的方法: setAge:, setName:, class, dealloc, _isKVOA, 

-----------------end------------

0x00000001083a3c60

0x00000001089807fa

0x00000001083a3c60

0x00000001089807fa

可以看到输出类名始终为:TestKVOKVC.People,这是因为新诞生的派生类重写了 -class 方法声称它就是起初的基类,只有使用 runtime 函数 object_getClass 才能一睹芳容:NSKVONotifying_TestKVOKVC.People。注意看:name,age 两个被观察对象真正的类型都是 NSKVONotifying_TestKVOKVC.People,而且该类实现了:setAge:, setName:, class, dealloc, _isKVOA 这些方法。其中 setAge:, setName:重写实现通知机制, class声明原来的类 , dealloc释放资源,私有方法 _isKVOA 是用来标示该类是一个 KVO 机制声称的类。在这里 swift 做了一些优化,它对所有被观察对象只生成一个派生类,该派生类实现所有被观察对象的 setter 方法,这样就减少了派生类的数量,提供了效率。所有 NSKVONotifying_TestKVOKVC.People 这个派生类重写了 setAge,setName方法(留意:没重写其他set方法)。

接着来看最后两行输出,地址 0x00000001083a3c60 是 TestKVOKVC.People 类中的实现,而地址是 0x00000001089807fa 是派生类 NSKVONotifying_TestKVOKVC.People 类中的实现。那后面那个地址到底是什么呢?可以通过 lldb 的 info 命令 image lookup --address  0x00000001089807fa 查看该地址的信息:

(lldb) image lookup --address  0x00000001089807fa

      Address: Foundation[0x00000000000637fa] (Foundation.__TEXT.__text + 401690)

      Summary: Foundation`_NSSetObjectValueAndNotify

 

看起来它是 Foundation 框架提供的私有函数:_NSSetObjectValueAndNotify。更进一步,我们来看看 Foundation 到底提供了哪些用于 KVO 的辅助函数。打开 terminal,使用 nm -a 命令查看 Foundation 中的信息:

nm -a /System/Library/Frameworks/Foundation.framework/Versions/C/Foundation

其中查找到我们关注的函数:

00000000000c76bf t __NSSetBoolValueAndNotify

0000000000152ab2 t __NSSetBoolValueForKeyInIvar

00000000000c74c8 t __NSSetBoolValueForKeyWithMethod

00000000000997db t __NSSetCharValueAndNotify

0000000000064759 t __NSSetCharValueForKeyInIvar

00000000000646f1 t __NSSetCharValueForKeyWithMethod

000000000001bd73 t __NSSetCheckSize

00000000003f7a50 b __NSSetClass

00000000001019af t __NSSetDirectory2

00000000000ab4d3 t __NSSetDoubleValueAndNotify

0000000000152b5d t __NSSetDoubleValueForKeyInIvar

00000000000abf65 t __NSSetDoubleValueForKeyWithMethod

0000000000139763 T __NSSetExceptionRaiser

0000000000180650 t __NSSetFileModificationUNIXTime

00000000000c9cca t __NSSetFloatValueAndNotify

0000000000152bb6 t __NSSetFloatValueForKeyInIvar

0000000000152712 t __NSSetFloatValueForKeyWithMethod

00000000000c9ba9 t __NSSetIntValueAndNotify

0000000000152c0f t __NSSetIntValueForKeyInIvar

00000000000abefb t __NSSetIntValueForKeyWithMethod

00000000001817a7 T __NSSetLogCStringFunction

00000000000a5389 t __NSSetLongLongValueAndNotify

0000000000152d5f t __NSSetLongLongValueForKeyInIvar

00000000000a55e1 t __NSSetLongLongValueForKeyWithMethod

000000000015abd7 t __NSSetLongValueAndNotify

0000000000152cb7 t __NSSetLongValueForKeyInIvar

0000000000152779 t __NSSetLongValueForKeyWithMethod

000000000011651f T __NSSetMainBundle

00000000001531ad t __NSSetNilValueForKey

00000000000647b0 t __NSSetObjectSetIvarValueForKeyInIvar

0000000000096e38 t __NSSetObjectValueAndNotify

0000000000028b2d t __NSSetObjectValueForKeyInIvar

00000000000c0657 t __NSSetPointValueAndNotify

0000000000152e5f t __NSSetPointValueForKeyInIvar

00000000000c05f1 t __NSSetPointValueForKeyWithMethod

000000000015b067 t __NSSetRangeValueAndNotify

0000000000152ebc t __NSSetRangeValueForKeyInIvar

00000000001528b2 t __NSSetRangeValueForKeyWithMethod

0000000000099610 t __NSSetRectValueAndNotify

0000000000152f15 t __NSSetRectValueForKeyInIvar

000000000015291d t __NSSetRectValueForKeyWithMethod

000000000015ae1f t __NSSetShortValueAndNotify

0000000000152db3 t __NSSetShortValueForKeyInIvar

0000000000152849 t __NSSetShortValueForKeyWithMethod

00000000000af692 t __NSSetSizeValueAndNotify

0000000000152f86 t __NSSetSizeValueForKeyInIvar

00000000000af62d t __NSSetSizeValueForKeyWithMethod

000000000015aab3 t __NSSetUnsignedCharValueAndNotify

0000000000152b09 t __NSSetUnsignedCharValueForKeyInIvar

00000000001526a9 t __NSSetUnsignedCharValueForKeyWithMethod

00000000000e59a2 t __NSSetUnsignedIntValueAndNotify

0000000000152c63 t __NSSetUnsignedIntValueForKeyInIvar

0000000000103f5e t __NSSetUnsignedIntValueForKeyWithMethod

0000000000096fa8 t __NSSetUnsignedLongLongValueAndNotify

00000000000f5f93 t __NSSetUnsignedLongLongValueForKeyInIvar

00000000000c745d t __NSSetUnsignedLongLongValueForKeyWithMethod

000000000015acfa t __NSSetUnsignedLongValueAndNotify

0000000000152d0b t __NSSetUnsignedLongValueForKeyInIvar

00000000001527e1 t __NSSetUnsignedLongValueForKeyWithMethod

000000000015af43 t __NSSetUnsignedShortValueAndNotify

0000000000152e09 t __NSSetUnsignedShortValueForKeyInIvar

0000000000103ef5 t __NSSetUnsignedShortValueForKeyWithMethod

 

Foundation 提供了大部分基础数据类型的辅助函数,此外还包括一些常见的 Cocoa 结构体如 Point, Range, Rect, Size,这表明这些结构体也可以用于自动键值观察,但要注意除此之外的结构体就不能用于自动键值观察了。

 

 

 

待续...........

 

 demo:https://github.com/easyui/TestKVO

 

 

 参考:

http://blog.csdn.net/wzzvictory/article/details/9674431?utm_source=tuicool

http://blog.csdn.net/kesalin/article/details/8194240

http://www.cnblogs.com/dark-angel/archive/2011/05/05/2037734.html

http://objccn.io/issue-7-3/#key-value-validation

http://www.cnblogs.com/496668219long/p/4470923.html

http://www.jianshu.com/p/e036e53d240e

http://blog.sina.com.cn/s/blog_8a38e5240100u2yw.html

http://www.androiddev.net/kvo/

http://nshipster.com/key-value-observing/

http://tech.glowing.com/cn/implement-kvo/

https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/KeyValueObserving/Articles/KVOImplementation.html

分享到:
评论
1 楼 roadProgram 2016-05-03  
文章很犀利

相关推荐

    Swift5.2 KVO

    Swift5.2 Key-Value Observing (KVO) 是一种Objective-C中的机制,但在Swift中也可以使用,主要是为了实现对象属性变化的监听。KVO允许一个对象观察另一个对象的某个属性,当该属性的值发生变化时,观察者会收到通知...

    swift-通过willMoveToSuperview的无忧的使用KVO

    在Swift编程中,KVO(Key-ValueObserving,键值观察)是一种强大的机制,用于监听对象属性的变化。本文将详细讲解如何在Swift中利用`willMoveToSuperview`方法无痛地实现KVO,以及其在Swift开发中的应用。 ### 1. ...

    ios KVO实现原理

    **iOS KVO(Key-Value Observing)实现原理** Key-Value Observing,简称KVO,是Objective-C中的一种机制,允许对象监听其他对象的属性变化。KVO是Apple为Foundation框架提供的一种强大的数据绑定技术,它使得一个...

    swift-KVC与KVO使用姿势和原理解析

    Swift中的Key-Value Coding(KVC)和Key-Value Observing(KVO)是两种强大的数据绑定技术,它们允许程序员间接访问对象的属性并监听这些属性的变化。在本文中,我们将深入探讨这两种技术的使用方法、工作原理以及...

    swift-一句话使用KVO使用完无需自己移除KVO

    在Swift编程中,Key-Value Observing(KVO)是一种观察者模式的实现,用于监听对象属性的变化。KVO在Objective-C中广泛使用,并且在Swift中也可以通过桥接头文件来实现。标题提到的“swift-一句话使用KVO使用完无需...

    KVO实现原理demo

    **KVO(Key-ValueObserving)实现原理详解** KVO,全称为Key-ValueObserving,是Objective-C中一种强大的数据绑定机制,允许对象监听并自动响应其他对象属性值的变化。KVO是Apple在Foundation框架中提供的一个API,...

    iOS KVO 监听frame

    在iOS开发中,Key-Value Observing(KVO)是一种强大的机制,允许对象监听其他对象属性的变化。在标题“iOS KVO 监听frame”中,我们关注的是如何使用KVO来跟踪和响应UI控件(如UIView)的frame属性变化。这在布局...

    KVC、kVO原理详解

    ### KVC、kVO原理详解 #### 一、KVC(Key-Value Coding)与 KVO(Key-Value Observing)简介 KVC 和 KVO 是 Objective-C 中两个非常重要的特性,它们为开发者提供了强大的数据访问和观察机制。这两种技术在 Cocoa ...

    swift-Ausbin-基于数据驱动的Swift框架(KVO)

    Swift-Ausbin是一个基于数据驱动的Swift框架,其核心特性是采用了Key-Value Observing (KVO)机制。KVO是Objective-C中的一个强大的特性,用于观察并响应对象属性值的变化。在Swift中,虽然没有原生支持KVO,但通过...

    KVC-KVO原理详解及编程指南

    详细的介绍了KVO与KVC的原理机制,可以有效运用KVO-KVC编程

    KVO机制工作原理

    **KVO机制工作原理** Key-Value Observing(KVO)是Objective-C中的一种机制,它允许对象监听并响应其他对象属性的变化。这种机制在iOS和macOS开发中广泛用于数据绑定和实时更新视图。KVO是Apple的Cocoa框架的一...

    Swift-KVO转换模型

    Swift-KVO,即Key-Value Observing,是iOS开发中的一种数据绑定技术,它允许对象监听其他对象的属性变化。在Objective-C中,KVO是一个强大的特性,而在Swift中,虽然原生不支持,但可以通过一些技巧实现。这篇文章将...

    swift-iOS安全的KVO操作解决由于添加了KVO忘记移除或者多次移除导致的崩溃问题

    首先,我们需要理解KVO的基本原理。KVO是Objective-C的一部分,但在Swift中也可以通过桥接头文件使用。当一个对象注册为另一个对象的观察者时,它会接收到被观察对象属性变化的通知。通常,我们使用`addObserver...

    Swift5.2 KVO监听

    Swift5.2中的Key-Value Observing(KVO)是一种观察者模式的实现,它允许一个对象监听并响应另一个对象的属性变化。在iOS、macOS等Apple平台的开发中,KVO是一种强大的机制,可以帮助我们实现数据绑定和动态更新视图...

    swift-Packageruntime实现的block形式的KVO

    在Swift编程语言中,Key-ValueObserving (KVO)是一种强大的机制,允许我们监听某个对象的属性变化。虽然Objective-C原生支持KVO,但Swift并没有直接提供类似的功能。然而,开发者可以通过自定义实现来达到类似的效果...

    swift-响应式编程思想KVO的底层实现

    在iOS开发中,Swift虽然原生并不支持响应式编程,但我们可以通过KVO(Key-Value Observing)或者第三方库如RxSwift、ReactiveCocoa来实现类似的功能。 KVO,即Key-Value Observing,是Objective-C中的一个特性,但...

    Swift-KVO+Runtime转换模型

    这个主题“Swift-KVO+Runtime转换模型”涉及到Swift中的Key-Value Observing (KVO)和Runtime机制,它们是Objective-C的特性,但在Swift中也可以通过桥接头文件使用。 **Key-Value Observing (KVO)** 是Objective-C...

    ios-KVO底层实现--利用runtime简单的实现KVO底层原理.zip

    2. **KVO基本原理** KVO的实现依赖于Objective-C的isa指针和消息发送机制。当一个对象被添加到观察者列表后,系统会动态地生成一个新的类,这个类继承自原始类,并重写了被观察属性的setter方法,以实现通知观察者...

    iOS开发·KVO用法,原理与底层实现: runtime模拟实现KVO监听机制(Blcok及Delgate方式).zip

    iOS开发·KVO用法,原理与底层实现: runtime模拟实现KVO监听机制(Blcok及Delgate方式).zip,iOS开发·KVO用法,原理与底层实现: runtime模拟实现KVO监听机制(Blcok及Delgate方式)

    swift-中心化的KVO简易的数据驱动

    正在开发的App为金融类型,后台频繁快速推送数据,UI界面需要及时响应。为了解决频繁reload UITableView等控件,解决办法是某个数据发生变化,仅刷新相应的独立UI元素(如,UILable,UIButton)。

Global site tag (gtag.js) - Google Analytics