UITableviewcell的性能問題

iOS開發UI篇—UITableviewcell的性能問題緩存

1、UITableviewcell的一些介紹ide

UITableView的每一行都是一個UITableViewCell,經過dataSource的 tableView:cellForRowAtIndexPath:方法來初始化每⼀行性能

UITableViewCell內部有個默認的子視圖:contentView,contentView是UITableViewCell所顯示內容的父視圖,可顯示一些輔助指示視圖優化

輔助指示視圖的做⽤是顯示一個表示動做的圖標,能夠經過設置UITableViewCell的 accessoryType來顯示,默認是UITableViewCellAccessoryNone(不顯⽰示輔助指⽰示視圖), 其餘值以下:atom

UITableViewCellAccessoryDisclosureIndicatorspa

UITableViewCellAccessoryDetailDisclosureButtoncode

UITableViewCellAccessoryCheckmark對象

還能夠經過cell的accessoryView屬性來自定義輔助指示視圖(⽐如往右邊放一個開關) blog

 

2、問題圖片

  cell的工做:在程序執行的時候,能看到多少條,它就建立多少條數據,若是視圖滾動那麼再建立新顯示的內容。(系統自動調用)。即當一個cell出如今視野範圍內的時候,就會調用建立一個cell。這樣的邏輯看上去沒有什麼問題,可是真的沒有任何問題嗎?

  當建立調用的時候,咱們使用nslog打印消息,並打印建立的cell的地址。咱們發現若是數據量很是大,用戶在短期內來回滾動的話,那麼會建立大量的cell,一直開闢空間,且若是是往回滾,經過打印地址,咱們會發現它並無重用以前已經建立的cell,而是從新建立,開闢新的存儲空間。

  那有沒有什麼好的解決辦法呢?

 

3、cell的重用原理

(1) iOS設備的內存有限,若是用UITableView顯示成千上萬條數據,就須要成千上萬 個UITableViewCell對象的話,那將會耗盡iOS設備的內存。要解決該問題,須要重用UITableViewCell對象

(2)重⽤原理:當滾動列表時,部分UITableViewCell會移出窗口,UITableView會將窗口外的UITableViewCell放入一個對象池中,等待重用。當UITableView要求dataSource返回 UITableViewCell時,dataSource會先查看這個對象池,若是池中有未使用的UITableViewCell,dataSource則會用新的數據來配置這個UITableViewCell,而後返回給 UITableView,從新顯示到窗口中,從而避免建立新對象 。這樣可讓建立的cell的數量維持在很低的水平,若是一個窗口中只能顯示5個cell,那麼cell重用以後,只須要建立6個cell就夠了。

(3)注意點:還有⼀個很是重要的問題:有時候須要自定義UITableViewCell(用⼀個子類繼 承UITableViewCell),並且每⼀行⽤的不必定是同一種UITableViewCell,因此一 個UITableView可能擁有不一樣類型的UITableViewCell,對象池中也會有不少不一樣類型的 UITableViewCell,那麼UITableView在重⽤用UITableViewCell時可能會獲得錯誤類型的 UITableViewCell

解決⽅方案:UITableViewCell有個NSString *reuseIdentifier屬性,能夠在初始化UITableViewCell的時候傳入一個特定的字符串標識來設置reuseIdentifier(通常用UITableViewCell的類名)。當UITableView要求dataSource返回UITableViewCell時,先 經過一個字符串標識到對象池中查找對應類型的UITableViewCell對象,若是有,就重用,若是沒有,就傳入這個字符串標識來初始化⼀一個UITableViewCell對象。

圖片示例:

 

說明:一個窗口放得下(可視)三個cell,整個程序只須要建立4個該類型的cell便可。

 

4、cell的優化代碼

 代碼示例:

複製代碼
 1 #import "NJViewController.h"
 2 #import "NJHero.h"
 3 
 4 // #define ID @"ABC"
 5 
 6 @interface NJViewController ()<UITableViewDataSource, UITableViewDelegate>
 7 /**
 8  *  保存全部的英雄數據
 9  */
10 @property (nonatomic, strong) NSArray *heros;
11 @property (weak, nonatomic) IBOutlet UITableView *tableView;
12 
13 @end
14 
15 @implementation NJViewController
16 
17 #pragma mark - 懶加載
18 - (NSArray *)heros
19 {
20     if (_heros == nil) {
21         // 1.得到全路徑
22         NSString *fullPath =  [[NSBundle mainBundle] pathForResource:@"heros" ofType:@"plist"];
23         // 2.更具全路徑加載數據
24         NSArray *dictArray = [NSArray arrayWithContentsOfFile:fullPath];
25         // 3.字典轉模型
26         NSMutableArray *models = [NSMutableArray arrayWithCapacity:dictArray.count];
27         for (NSDictionary *dict in dictArray) {
28             NJHero *hero = [NJHero heroWithDict:dict];
29             [models addObject:hero];
30         }
31         // 4.賦值數據
32         _heros = [models copy];
33     }
34     // 4.返回數據
35     return _heros;
36 }
37 
38 - (void)viewDidLoad
39 {
40     [super viewDidLoad];
41     // 設置Cell的高度
42     // 當每一行的cell高度一致的時候使用屬性設置cell的高度
43     self.tableView.rowHeight = 160;
44 }
45 
46 #pragma mark - UITableViewDataSource
47 // 返回多少組
48 - (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
49 {
50     return 1;
51 }
52 // 返回每一組有多少行
53 - (NSInteger) tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
54 {
55     return self.heros.count;
56 }
57 // 當一個cell出現視野範圍內的時候就會調用
58 // 返回哪一組的哪一行顯示什麼內容
59 - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
60 {
61     // 定義變量保存重用標記的值
62     static NSString *identifier = @"hero";
63     
64 //    1.先去緩存池中查找是否有知足條件的Cell
65     UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
66 //    2.若是緩存池中沒有符合條件的cell,就本身建立一個Cell
67     if (cell == nil) {
68         //    3.建立Cell, 而且設置一個惟一的標記
69         cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:identifier];
70         NSLog(@"建立一個新的Cell");
71     }
72 //    4.給cell設置數據
73     NJHero *hero = self.heros[indexPath.row];
74     cell.textLabel.text = hero.name;
75     cell.detailTextLabel.text = hero.intro;
76     cell.imageView.image = [UIImage imageNamed:hero.icon];
77     
78    //  NSLog(@"%@ - %d - %p", hero.name, indexPath.row, cell);
79     
80     // 3.返回cell
81     return cell;
82 }
83 
84 #pragma mark - 控制狀態欄是否顯示
85 /**
86  *   返回YES表明隱藏狀態欄, NO相反
87  */
88 - (BOOL)prefersStatusBarHidden
89 {
90     return YES;
91 }
92 @end
複製代碼

緩存優化的思路:

(1)先去緩存池中查找是否有知足條件的cell,如有那就直接拿來

(2)若沒有,就本身建立一個cell

(3)建立cell,而且設置一個惟一的標記(把屬於「」的給蓋個章)

(4)給cell設置數據

注意點:

定義變量用來保存重用標記的值,這裏不推薦使用宏(#define來處理),由於該變量只在這個做用域的內部使用,且若是使用宏定義的話,定義和使用位置太分散,不利於閱讀程序。因爲其值不變,沒有必要每次都開闢一次,因此用static定義爲一個靜態變量。

相關文章
相關標籤/搜索