通常對于相對復(fù)雜界面的狀態(tài)的控制我們都會選擇KVO,其中對數(shù)組的檢測相對于其他的數(shù)據(jù)會有點不一樣。
</br>
KVO概述
Key-Value Observing,它提供一種機制,當指定的對象的屬性被修改后,則對象就會接受到通知。簡單的說就是每次指定的被觀察的對象的屬性被修改后,KVO就會自動通知相應(yīng)的觀察者了。
不過我覺得用指針來形容會更加地確切:
當指定的對象的屬性指向的地址(不包括地址內(nèi)容修改)被修改后,則對象就會接受到通知。簡單的說就是每次指定的被觀察的對象的屬性指向的地址被修改后,KVO就會自動通知相應(yīng)的觀察者了。
<p>
所以對與一般的NSArry、NSString類型等,他們直接在修改的是指向的地址空間,所以可以直接設(shè)置kvo。
</p>
<p>
但是如NSMutableString、NSMutableArry等,他們修改的是指向地址的內(nèi)容,指向的地址空間并沒有發(fā)生改變。所以不能簡單地設(shè)置一個kvo監(jiān)聽器。說到這里,大家也肯定明白應(yīng)該如果修改了,沒錯,就是創(chuàng)建一個新的對象!
</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"];
而應(yīng)該使用[[self mutableArrayValueForKey:@"arry"] addObject:@"4"];
</br>
需要注意的是當一個object設(shè)置了監(jiān)聽者時,不能去修改object,以免造成crash
類似的很多情況都可以在內(nèi)存的角度得到解決,希望能夠幫到您!
補充最近項目中遇到的問題,當使用KVO監(jiān)聽tableview的row數(shù)量的時候(一般使用一個NSArryData數(shù)組)。這個時候?qū)ell進行刪除操作,一般情況下是需要先對數(shù)據(jù)進行刪除,然后再對cell的進行刪除。但這邊crash。最后發(fā)現(xiàn)需要將數(shù)據(jù)的刪除操作放在cell的刪除之后。
[tableView beginUpdates];
[weakSelf.tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];
[[weakSelf.viewModel mutableArrayValueForKey:@"data"] removeObjectAtIndex:indexPath.row];//刪數(shù)據(jù)
[tableView endUpdates];
[tableView reloadData];