ios关于数组的KVO

通常对于相对复杂界面的状态的控制我们都会选择KVO,其中对数组的检测相对于其他的数据会有点不一样。



KVO概述

Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,则对象就会接受到通知。简单的说就是每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者了。

不过我觉得用指针来形容会更加地确切:

当指定的对象的属性指向的地址(不包括地址内容修改)被修改后,则对象就会接受到通知。简单的说就是每次指定的被观察的对象的属性指向的地址被修改后,KVO就会自动通知相应的观察者了。


所以对与一般的NSArry、NSString类型等,他们直接在修改的是指向的地址空间,所以可以直接设置kvo。



但是如NSMutableString、NSMutableArry等,他们修改的是指向地址的内容,指向的地址空间并没有发生改变。所以不能简单地设置一个kvo监听器。说到这里,大家也肯定明白应该如果修改了,没错,就是创建一个新的对象!

下面我们以NSMutableArry为例子

#import "MemoryViewController.h"

@interface MemoryViewController ()
@property (nonatomic, strong) NSMutableArray *arry;

@end

@implementation MemoryViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    
    self.view.backgroundColor = [UIColor yellowColor];
    
    self.arry = [NSMutableArray arrayWithObjects:@"1", @"2", @"3", nil];
    NSLog(@"arry.count: %ld, arry:%p", _arry.count, _arry);

    
    [self addObserver:self forKeyPath:@"arry" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:nil];
}

- (void)dealloc{
    [self removeObserver:self forKeyPath:@"arry"];
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event{
    [[self mutableArrayValueForKey:@"arry"] addObject:@"4"];
}

#pragma mark - kvo

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context{
    
    NSLog(@"keyPath: %@", keyPath);
}

这里我们不能再直接使用 [self.arry addObject:@"4"];
而应该使用[[self mutableArrayValueForKey:@"arry"] addObject:@"4"];



需要注意的是当一个object设置了监听者时,不能去修改object,以免造成crash

类似的很多情况都可以在内存的角度得到解决,希望能够帮到您!

补充最近项目中遇到的问题,当使用KVO监听tableview的row数量的时候(一般使用一个NSArryData数组)。这个时候对cell进行删除操作,一般情况下是需要先对数据进行删除,然后再对cell的进行删除。但这边crash。最后发现需要将数据的删除操作放在cell的删除之后。


 [tableView beginUpdates];
 [weakSelf.tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
 [[weakSelf.viewModel mutableArrayValueForKey:@"data"] removeObjectAtIndex:indexPath.row];//删数据
 [tableView endUpdates];
 [tableView reloadData];

你可能感兴趣的:(ios关于数组的KVO)