通常对于相对复杂界面的状态的控制我们都会选择KVO,其中对数组的检测相对于其他的数据会有点不一样。
</br>
KVO概述
Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,则对象就会接受到通知。简单的说就是每次指定的被观察的对象的属性被修改后,KVO就会自动通知相应的观察者了。
不过我觉得用指针来形容会更加地确切:
当指定的对象的属性指向的地址(不包括地址内容修改)被修改后,则对象就会接受到通知。简单的说就是每次指定的被观察的对象的属性指向的地址被修改后,KVO就会自动通知相应的观察者了。
<p>
所以对与一般的NSArry、NSString类型等,他们直接在修改的是指向的地址空间,所以可以直接设置kvo。
</p>
<p>
但是如NSMutableString、NSMutableArry等,他们修改的是指向地址的内容,指向的地址空间并没有发生改变。所以不能简单地设置一个kvo监听器。说到这里,大家也肯定明白应该如果修改了,没错,就是创建一个新的对象!
</p>
下面我们以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<UITouch *> *)touches withEvent:(UIEvent *)event{
[[self mutableArrayValueForKey:@"arry"] addObject:@"4"];
}
#pragma mark - kvo
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
NSLog(@"keyPath: %@", keyPath);
}
这里我们不能再直接使用 [self.arry addObject:@"4"];
而应该使用[[self mutableArrayValueForKey:@"arry"] addObject:@"4"];
</br>
需要注意的是当一个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];