1.簡述
在IM通訊中或者直播中,消息列表界面需要經常進行刷新, 如果每次接收到數據都刷新一次,有時在短時間內信息量太大的話,可能會造成刷新的過于頻繁,為了解決這個問題,可以嘗試下面的方法.
2.思路
可以通過控制器監聽runloop的運行狀態, 當runloop空閑的時候, 再進行UI的刷新. 并在刷新之前,設置好時間間隔,以及每次刷新的個數, 從而提升性能.
3.實現方案
3.1 一次性刷新方案(IM)
@interface ChatVC ()
// runloop觀察者
@property (nonatomic, assign) CFRunLoopObserverRef observerRef;
// 是否需要刷新列表
@property (nonatomic, assign) BOOL isNeedToRefreshChatList;
@end
- (void)viewDidLoad {
[self runloopObserver];
[NSNotificationCenter.defaultCenter addObserver:self selector:@selector(needToRefreshChatList) name:kRefreshChatListNoti object:nil];
}
- (void)dealloc {
[NSNotificationCenter.defaultCenter removeObserver:self];
// 移除runloop的監聽
CFRunLoopRemoveObserver(CFRunLoopGetCurrent(), self.observerRef, kCFRunLoopCommonModes);
NSLog(@"ChatVC dealloc");
}
// 刷新列表的通知方法
- (void)needToRefreshChatList {
// 標記列表需要刷新
self.isNeedToRefreshChatList = YES;
// 喚醒runloop
CFRunLoopWakeUp(CFRunLoopGetMain());
}
// 創建runloop監聽
- (void)runloopObserver{
@weakify(self)
__block NSTimeInterval timeInterVal = [[NSDate date] timeIntervalSince1970];
// 獲取到runloop的觀察者
self.observerRef = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
if (activity == kCFRunLoopBeforeWaiting) { // runloop空閑的時候刷新需要處理的列表
// 控制刷新頻率
NSTimeInterval currentTimeInterval = [[NSDate date] timeIntervalSince1970];
if (currentTimeInterval - timeInterVal < 0.1) {
return;
}
timeInterVal = currentTimeInterval;
if (weak_self.isNeedToRefreshChatList && weak_self.tableView) { // 判斷是否需要刷新
// 處理列表的刷新
[weak_self refreshChatList];
// 標記刷新完畢,無需重復刷新
weak_self.isNeedToRefreshChatList = NO;
}
}
});
// 添加監聽, 模式為 kCFRunLoopCommonModes
CFRunLoopAddObserver(CFRunLoopGetCurrent(), self.observerRef, kCFRunLoopCommonModes);
}
3.2每次最多刷新10條數據,直到沒有新的數據, 不在刷新(直播)
// 思路: 需要一個全局數組,存儲所有的信息, 每次抽取其中的10條信息, 添加到tableview的數據源中, 并從數組中移除這10條數據, 刷新tableview進行展示
// 創建runloop 監聽
- (void)runloopObserver{
__weak typeof(self) weakSelf = self;
__block NSTimeInterval timeInterVal = [[NSDate date] timeIntervalSince1970];
observerRef = CFRunLoopObserverCreateWithHandler(CFAllocatorGetDefault(), kCFRunLoopAllActivities, YES, 0, ^(CFRunLoopObserverRef observer, CFRunLoopActivity activity) {
NSLog(@"當前線程是: %@", [NSThread currentThread]);
// 設置刷新頻率 最少0.5秒刷新一次
//來處理數據量 (直播刷留言)
if (activity == kCFRunLoopBeforeWaiting) {
// 控制頻率
NSTimeInterval currentTimeInterVal = [[NSDate date] timeIntervalSince1970];
if (currentTimeInterVal - timeInterVal < 0.2) {
return ;
}
timeInterVal = currentTimeInterVal;
// 處理數據
[weakSelf.dataLock lock];
NSArray *subArr = nil;
if (weakSelf.newsDataArr.count > 0) { // 一次最多拿10條數據
// 頻率可以通過newsDataArr的數據量來控制
NSRange range;
if (weakSelf.newsDataArr.count >= 10) {
range = NSMakeRange(0, 10);
}else{
range = NSMakeRange(0, weakSelf.newsDataArr.count-1);
}
subArr = [weakSelf.newsDataArr subarrayWithRange:range];
[weakSelf.newsDataArr removeObjectsInRange:range];
}
[weakSelf.dataArr addObjectsFromArray:subArr];
[weakSelf.dataLock unlock];
[weakSelf.tableView reloadData]; // 刷新界面
[weakSelf.tableView layoutIfNeeded];
[weakSelf.tableView setContentOffset:CGPointMake(0, weakSelf.tableView.contentSize.height - weakSelf.tableView.frame.size.height)];
}
});
CFRunLoopAddObserver(CFRunLoopGetCurrent(), observerRef, kCFRunLoopCommonModes);
}