基础 (十五) : UITableView

掌握

设置UITableView的dataSource、delegate

UITableView多组数据和单组数据的展示

UITableViewCell的常见属性

UITableView的性能优化(cell的循环利用)

自定义Cell

什么是UITableView?

能看到各式各样的表格数据就是UITableView

1.png
2.png

如何展示数据

UITableView需要一个数据源(dataSource)来显示数据

注意:但是如果删掉原先控制器自动配置的View,用自己定义的UITableView,系统是默认遵守数据源和协议,不用设置!

UITableView会向数据源查询一共有多少行数据以及每一行显示什么数据等

没有设置数据源的UITableView只是个空壳

凡是遵守UITableViewDataSource协议的OC对象,都可以是UITableView的数据源

/**

 *  dataSource是UITableViewDataSource类型,主要为UITableView提供显示用的数据(UITableViewCell),指定UITableViewCell支持的编辑操作类型(insert,delete和reordering),并根据用户的操作进行相应的数据更新操作,如果数据没有更具操作进行正确的更新,可能会导致显示异常,甚至crush。




 delegate是UITableViewDelegate类型,主要提供一些可选的方法,用来控制tableView的选择、指定section的头和尾的显示以及协助完成cell的删除和排序等功能。




 提到UITableView,就必须的说一说NSIndexPath。UITableView声明了一个NSIndexPath的类别,主要用来标识当前cell的在tableView中的位置,该类别有section和row两个属性,前者标识当前cell处于第几个section中,后者代表在该section中的第几行。




 UITableView只能有一列数据(cell),且只支持纵向滑动,当创建好的tablView第一次显示的时候,我们需要调用其reloadData方法,强制刷新一次,从而使tableView的数据更新到最新状态。

*/


3.png

tableView展示数据的过程

调用数据源的下面方法得知一共有多少组数据

-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;

调用数据源的下面方法得知每一组有多少行数据

-(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section;

调用数据源的下面方法得知每一行显示什么内容

  • (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath;
4.png

初始MVC

MVC是一种设计思想,贯穿于整个iOS开发中,需要积累一定的项目经验,才能深刻体会其中的含义和好处

MVC中的三个角色

M:Model,模型数据

V:View,视图(界面)

C:Control,控制中心

注意:首先控制器会加载这个数据,然后根据这个数据的个数去创建对应的view

MVC的几个明显的特征和体现:

View上面显示什么东西,取决于Model

只要Model数据改了,View的显示状态会跟着更改

Control负责初始化Model,并将Model传递给View去解析展示

Cell简介以及属性

UITableView的每一行都是一个UITableViewCell,通过dataSource的tableView:cellForRowAtIndexPath:(此方法就是UITableViewCell重用机制的一个初始化Cell的一个系统提供的一个方法)方法来初始化每一行

UITableViewCell内部有个默认的子视图:contentView,contentView是UITableViewCell所显示内容的父视图,可显示一些辅助指示视图

accessoryType属性:

辅助指示视图的作用是显示一个表示动作的图标,可以通过设置UITableViewCell的accessoryType(其属性)来显示,默认是UITableViewCellAccessoryNone(不显示辅助指示视图),其他值如下:

属性accessoryType就是设置指示器,也就是手机右边的一个箭头告诉用户有的是可以点击的,以下皆是其枚举值

UITableViewCellAccessoryDisclosureIndicator
箭头

UITableViewCellAccessoryDetailDisclosureButton按钮 + 箭头

UITableViewCellAccessoryCheckmark 对勾

UITableViewCellAccessoryNone 没有

UITableViewCellAccessoryDetaiButton 叹号按钮

Cell的常见属性

设置cell的背景颜色 backgroundColor优先级低于backgroundView

backgroundView使用方法:

创建一个View

UIView *bg = [[UIView alloc] init];

设置view的颜色,然后简介的给cell赋值

bg.backgroundColor = [UIColor redColor];

再设置cell背景的颜色

cell.backgroundView= bg;

selectedBackgroundView使用方法:

创建一个View(也可以设置UIImageView作为cell的背景颜色)

UIView *selectedView= [[UIView alloc] init];

设置view的颜色,然后间接的赋值给cell

selectedView.backgroundColor = [UIColor redColor];

最后设置cell的背景颜色

cell.selectedBackgroundView= sele

还可以通过cell的accessoryView属性来自定义辅助指示视图(比如往右边放一个开关)

注意:这里的cell的指的是通过UITableViewCell创建的一个对象的accessoryView属性来自定义辅助的视图

Cell.accessoryView= UITableViewCellAccessoryCheckmark;

cell.accessoryView = [UIbutton alloc]init];(可以在右边自定义控件可以放一个按钮或者一个开关)

5.png

注意:以上cell的属性包含了4个属性

Cell的重用原理

iOS设备的内存有限,如果用UITableView显示成千上万条数据,就需要成千上万个UITableViewCell对象的话,那将会耗尽iOS设备的内存。要解决该问题,需要重用UITableViewCell对象

重用原理:当滚动列表时,部分UITableViewCell会移出窗口,UITableView会将窗口外的UITableViewCell放入一个对象池中,等待重用。当UITableView要求dataSource返回UITableViewCell时,dataSource会先查看这个对象池,如果池中有未使用的UITableViewCell,dataSource会用新的数据配置这个UITableViewCell,然后返回给UITableView,重新显示到窗口中,从而避免创建新对象

还有一个非常重要的问题:有时候需要自定义UITableViewCell(用一个子类继承UITableViewCell),而且每一行用的不一定是同一种UITableViewCell,所以一个UITableView可能拥有不同类型的UITableViewCell,对象池中也会有很多不同类型的UITableViewCell,那么UITableView在重用UITableViewCell时可能会得到错误类型的UITableViewCell

解决方案:UITableViewCell有个NSString
*reuseIdentifier属性,可以在初始化UITableViewCell的时候传入一个特定的字符串标识来设置reuseIdentifier(一般用UITableViewCell的类名)。当UITableView要求dataSource返回UITableViewCell时,先通过一个字符串标识到对象池中查找对应类型的UITableViewCell对象,如果有,就重用,如果没有,就传入这个字符串标识来初始化一个UITableViewCell对象

Cell的重用代码

(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath

{

注意:重用机制是根据相同的标识符来重用cell,标识符不同的cell不能彼此重用

1.定义一个cell的标识为ID

static NSString *ID = @"mjcell";




 2.从缓存池中取出cell

  UITableViewCell*cell = [tableView dequeueReusableCellWithIdentifier:ID];




 3.如果缓存池中没有cell


if (cell ==nil) {

    cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle

reuseIdentifier:ID];

}

 4.设置cell的属性...





return cell;

}

方法2:注册

      另一中性能优化的写法:

      0.先确定cell的重用标识

      1.注册带重用标识的cell

      2.从缓存池中取是否有带重用标识的cell(如果没有,系统会根据注册自动创建一个相应的cell返回给我们)

      3.覆盖cell上面的数据

      说明:当cell离开屏幕的时候,就会放到tableView的缓存池中,这时候缓存池才有数据


  



         3.0 先确定cell的重用标识:命名规范:cell类型+ID

        static NSString *ID = @"carID"; // static修饰局部变量,局部变量从执行后始终存在,

但不能被其它函数使用, 当再次进入该函数时, 将保存上次的结果。其它与局部变量一样。

  static BOOL isReged =NO;

         3.1 先注册带ID的cell

 
  if (isReged ==NO) {

warning registerCell需要注意的3点,面试可能会问到

该方法是伴随着UICollectionView出现的,也就是iOS6出现的

采用注册创建出来的cell,默认是Default样式,所以一般注册大部分都用在自定义cell的时候

需要注意的是 [tableView registerNib:<#(UINib *)#>
forCellReuseIdentifier:<#(NSString *)#>] 是iOS5出现的

     编码规范:xib自定义cell一般用注册

 
          [tableView

registerClass:[UITableViewCell class] forCellReuseIdentifier:ID];

          isReged =YES;


             

 
      }


    

         3.2 查看缓存池中是否有带重用标识的cell

         缓存池方法中封装了,如果缓存池中没有,就根据注册创建新的cell,然后返回给我们一个带ID的cell

 
  UITableViewCell *cell = [tableView

dequeueReusableCellWithIdentifier:ID];

         3.3 覆盖cell上的数据

 
  XMGCar *car = self.cars[indexPath.row];


    

 
  cell.textLabel.text = [NSString...


                 
                 
             


      3.4返回cell

 
  return cell;


                 
                
              }

方法3:storyboard

              另一中性能优化的写法(storyboard):


           

              0.先确定cell的重用标识

              1.将storyboard中的tableView中的cell的重用标志赋值

              2.从缓存池中取是否有带重用标识的cell

              (2.1

如果没有,系统会根据注册自动创建一个相应的cell返回给我们;

              2.2 如果也没有注册过,系统会根据storyboard中写好的带重用标志的cell来自动创建,然后返回)

              3.覆盖cell上面的数据

              说明:当cell离开屏幕的时候,就会放到tableView的缓存池中,这时候缓存池才有数据

3.返回每行内容:该方法,只有当cell显示在屏幕上的时候才会调用,是一种懒加载的思想

warning 如果有注册,就不会触发storyboard的创建新cell的机制.只有没有注册过,并且storyboard中没有标记相应cell 的时候,dequeueReusableCellWithIdentifier才会返回nil

- (UITableViewCell *)tableView:(UITableView *)tableView

cellForRowAtIndexPath:(NSIndexPath *)indexPath

{

             3.0 先确定cell的重用标识:命名规范:数据的模型+ID


          static NSString *ID

= @"carID"; static修饰局部变量,局部变量只会分配一次内存地址

             3.1 查看缓存池中是否有带重用标识的cell

             缓存池方法中封装了,如果缓存池中没有,就根据注册创建新的cell,然后返回给我们一个带ID的cell

             后面还封装了一层,如果也没有注册呢,会根据storyboard中是否标记了在重用cell,来创建一个新的cell然后返回

 
      UITableViewCell *cell = [tableView

dequeueReusableCellWithIdentifier:ID];

             3.3 覆盖cell上的数据

 
      XMGCar *car = self.cars[indexPath.row];

 
      cell.textLabel.text = [N...


                 
                 
                   



                 
                
                   



       return

cell;

 }


                 
                
                   












 3.修改控件属性(一半在cell的创建方法中设置initWithStyle)

cell.imageView.layer.cornerRadius= 22; 

设置圆角为正方形边长一半(内切圆半径)

cell.imageView.layer.masksToBounds= YES; 设置将位于imageView之下的layer都遮盖住(默认是NO)  等效代码是:cell.imageView.clipsToBounds



 设置字体以及文字颜色

cell.textLabel.textColor= [UIColor brownColor];

cell.detailTextLabel.font= [UIFont

systemFontOfSize:12.0];

cell.detailTextLabel.textColor= [UIColor grayColor];



 设置辅助视图

cell.accessoryType= UITableViewCellAccessoryDetailButton;

使用xib封装一个view的步骤

新建一个xib文件描述一个view的内部结构(假设叫做MJTgCell.xib)

新建一个自定义的类

(自定义类需要继承自系统自带的view, 继承自哪个类, 取决于xib根对象的Class)

新建类的类名最好跟xib的文件名保持一致(比如类名就叫做MJTgCell)

将xib中的控件 和 自定义类的.m文件 进行连线

提供一个类方法返回一个创建好的自定义view(屏蔽从xib加载的过程)

提供一个模型属性让外界传递模型数据

重写模型属性的setter方法,在这里将模型数据展示到对应的子控件上面

Delegate的使用场合

对象A内部发生了一些事情,想通知对象B

对象B想监听对象A内部发生了什么事情

对象A想在自己的方法内部调用对象B的某个方法,并且对象A不能对对象B有耦合依赖

对象A想传递数据给对象B

……

以上情况,结果都一样:对象B是对象A的代理(delegate)

使用delegate的步骤

先搞清楚谁是谁的代理(delegate)

定义代理协议,协议名称的命名规范:控件类名+ Delegate

定义代理方法

代理方法一般都定义为@optional

代理方法名都以控件名开头

代理方法至少有1个参数,将控件本身传递出去

设置代理(delegate)对象 (比如myView.delegate = xxxx;)

代理对象遵守协议

代理对象实现协议里面该实现的方法

在恰当的时刻调用代理对象(delegate)的代理方法,通知代理发生了什么事情

(在调用之前判断代理是否实现了该代理方法)

通过代码自定义cell(cell的高度不一致)

1.新建一个继承自UITableViewCell的类

注意:一个是初始化方法一个设置子控件的方法一个是设置数据的对象的set方法

2.重写initWithStyle:reuseIdentifier:方法

添加所有需要显示的子控件(不需要设置子控件的数据和frame, 子控件要添加到contentView中

进行子控件一次性的属性设置(有些属性只需要设置一次, 比如字体\固定的图片等等但是设置frame不行)

3.提供2个模型

数据模型: 存放文字数据\图片数据
(字典转模)

frame模型: 存放数据模型\所有子控件的frame\cell的高度

4.cell拥有一个frame模型(不要直接拥有数据模型)

5.重写frame模型属性的setter方法: 在这个方法中设置子控件的显示数据和frame

6.frame模型数据的初始化已经采取懒加载的方式(每一个cell对应的frame模型数据只加载一次)

UITextField

通过UITextField的代理方法能够监听键盘最右下角按钮的点击

成为UITextField的代理

self.textField.delegate = self;

遵守UITextFieldDelegate协议,实现代理方法

(BOOL)textFieldShouldReturn:(UITextField *)textField;

在UITextField左边放一个view

self.textField.leftView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 8, 0)];

self.textField.leftViewMode = UITextFieldViewModeAlways;

***************************笔记************************


tableView常见属性

修改cell的高度,也就是行高

self.tableView.rowHeight= 60;



 修改组头.组尾的高度(Plain样式下生效): 在Group样式下,组头组尾高度都是系统默认的

self.tableView.sectionHeaderHeight= 2;

self.tableView.sectionFooterHeight= 222;



 设置tableView的表头表尾视图

 一般用于做下拉刷新,banner

self.tableView.tableHeaderView= [UIButton

buttonWithType:UIButtonTypeContactAdd];

self.tableView.tableFooterView= [[UISwitch alloc] init];



 设置分割线颜色:

 [UIColor clearColor]如果设置成透明色就表示分割线取消显示

warning 在tableView上,尽量避免使用透明色

self.tableView.separatorColor= [UIColor orangeColor];



 UITableViewCellSeparatorStyleNone 取消分割线的显示

warning
UITableViewCellSeparatorStyleNone->UITableViewCellSelectionStyleNone

self.tableView.separatorStyle= UITableViewCellSeparatorStyleNone;

隐藏状态栏

  • (BOOL)prefersStatusBarHidden

{

return YES;

}

 设置cell的属性

 设置cell的背景颜色 backgroundColor优先级低于backgroundView

cell.backgroundColor= [UIColor colorWithRed:arc4random_uniform(255)/255.0 green:arc4random_uniform(255)/255.0 blue:arc4random_uniform(255)/255.0 alpha:1];



UIView*bg = [[UIView alloc] init];

bg.backgroundColor= [UIColor grayColor];

cell.backgroundView

= bg; 一般用于UIIMageView展示公司logo

typedef NS_ENUM(NSInteger,

UITableViewCellAccessoryType)

{

UITableViewCellAccessoryNone,    
           don't show any

accessory view 默认没有

UITableViewCellAccessoryDisclosureIndicator, regular chevron. doesn't track 尖尖

UITableViewCellAccessoryDetailDisclosureButton, info button w/ chevron. tracks 圈! + 尖尖

UITableViewCellAccessoryCheckmark,
checkmark. doesn't track 对号

UITableViewCellAccessoryDetailButton NS_ENUM_AVAILABLE_IOS(7_0) // info button. tracks 圈!

 设置辅助视图

 accessoryView优先级高

NSUIntegerrow = arc4random_uniform(255);

if (row % 2) {

    cell.accessoryView = [[UISwitch alloc] init];

}else

{

    cell.accessoryType = UITableViewCellAccessoryDetailDisclosureButton;

}

tableView如何展示数据?

mutiSectionsTableView

tableView如何展示数据?

  • 通过数据源

@property (nonatomic,
assign) id
<UITableViewDataSource> dataSource;

  • 设置数据源

1.在viewDidLoad方法中设置

tableView.dataSource =
self;

2.在storyboard中拖线

  • 遵守数据源协议<UITableViewDataSource>

  • 实现数据源方法

多少组数据

(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;

每一组有多少行数据

(NSInteger)tableView:(UITableView *)tableView
numberOfRowsInSection:(NSInteger)section;

每一行显示什么内容(note:必须是UITableViewCell或者其子类)

  • (UITableViewCell
    *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath
    *)indexPath;

indexPaht.section 第几组

indexPath.row 第几行

每一组的头部section 代表行数

  • (NSString
    *)tableView:(UITableView *)tableView
    titleForHeaderInSection:(NSInteger)section;

每一组的尾部

  • (NSString
    *)tableView:(UITableView *)tableView titleForFooterInSection:(NSInteger)section

为什么要数据独立?

新需求再添加一组数据时候,需要修改全部数据源方法!

思路:将所有需要修改的数据集中到一处

  • 创建一个新的数组属性

@property (nonatomic, strong) NSArray groups; /*< group模型数组 */

  • 通过懒加载为数组赋值

懒加载

  • (NSArray *)groups

{

if (!_groups) {

XMGGroup *g0 = [[XMGGroup alloc] init];

g0.header = @"adc";

  g0.footer = @"物理输出脆皮";


  g0.heros = @[[XMGHero heroWithName:@"艾希" Icon:@"hanbing"],


              [XMGHero

heroWithName:@"崔斯特娜" Icon:@"xiaopao"],

              [XMGHero

heroWithName:@"金克斯" Icon:@"金克斯"]];

XMGGroup *g1 = [[XMGGroup alloc] init];

g1.header = @"apc";

  g1.footer = @"法术输出脆皮";


  g1.heros = @[[XMGHero heroWithName:@"阿狸" Icon:@"ali"],


              [XMGHero

heroWithName:@"火男" Icon:@"huonan"],

              [XMGHero

heroWithName:@"球女" Icon:@"qiunv"]];

XMGGroup *g2 = [[XMGGroup alloc] init];

  g2.header = @"上单";


  g2.footer = @"上单爸爸不好惹";


  g2.heros = @[[XMGHero heroWithName:@"大保健" Icon:@"dabaojian"],


            ];

_groups = @[g0, g1, g2];

}

return _groups;

}

这样再添加数据,只需要对懒加载中的数据进行操作即可

实际开发中,所用的数据也是从服务器拿到后展示

分析模型的创建

  • 组头

  • 组尾

  • cell上的内容(数组)

  • 每个cell上的内容分为name和icon

  • 创建hero模型

单组数据展示

  • 数据源方法:返回多少组可以不实现
  • cell上自带3个展示信息的子控件

cell.textLabel.text =
car.name;

系统自带的cell的非value2样式并且有数据才展示

cell.imageView.image =
[UIImage imageNamed:car.icon];

系统自带cell的非default样式并且有数据时候才展示

cell.detailTextLabel.text
= car.money;

  • 复习简单plist文件解析

1.拿到plist的文件路径

NSString *filePath = [[NSBundle mainBundle]
pathForResource:@"cars.plist"ofType:nil];

2.创建对应的JSON数据

NSArray *dicts =
[NSArray arrayWithContentsOfFile:filePath];

3.JSON->Model

NSMutableArray
*arrayM = [NSMutableArray arrayWithCapacity:dicts.count];

for (NSDictionary *dict in
dicts) {

3.1 Dict->Model

XMGCar

*obj = [XMGCar carWithDict:dict];

[arrayM

addObject:obj];

}

  • Dict->Model
  • (instancetype)carWithDict:(NSDictionary *)dict

{

XMGCar

*car = [[self alloc] init];

   car.name = dict[@"name"];

car.icon = dict[@"icon"];

car.money = dict[@"money"];

 kvc 相当于上面3行

[car

setValuesForKeysWithDictionary:dict];

return car;

}

pragma mark - cell常见属性

方法抽取的技巧就是将变量写成参数传进来

相同业务逻辑的的代码尽量抽取成方法

  • (void)setupCellProperty:(UITableViewCell *)cell
    row:(NSInteger)row

{

 cell的常见属性



 设置cell的背景颜色

cell.backgroundColor = (row % 2)? [UIColor yellowColor]: [UIColor orangeColor];

 设置cell的背景视图,并且backgroundView的优先级比backgroundColor高一些

UIView *cellBg = [[UIView alloc] init];

cellBg.backgroundColor = [UIColor grayColor];

cell.backgroundView = cellBg;

/*

UITableViewCellAccessoryNone,
// don't show any accessory view

 UITableViewCellAccessoryDisclosureIndicator, 

// 尖尖regular chevron. doesn't track

 UITableViewCellAccessoryDetailDisclosureButton, // 圈i
  • 尖尖info button w/ chevron. tracks
 UITableViewCellAccessoryCheckmark,   
      // 小对勾checkmark. doesn't track

 UITableViewCellAccessoryDetailButton

NS_ENUM_AVAILABLE_IOS(7_0) // 圈i info button. tracks

*/

 cell的辅助视图样式

cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator;

 cell的辅助视图,作用优先级高于accessoryType

cell.accessoryView = [[UISwitch alloc] init];

/*

 UITableViewCellSelectionStyleNone, // 设置没有选中样式




 UITableViewCellSelectionStyleBlue, // 在iOS6之前,选中背景色变蓝,iOS7之后,和gray相同




 UITableViewCellSelectionStyleGray, // 默认样式

 UITableViewCellSelectionStyleDefault

NS_ENUM_AVAILABLE_IOS(7_0) // 默认样式

*/

cell.selectionStyle = UITableViewCellSelectionStyleGray;

 在UITableViewCellSelectionStyleNone样式中,设置selectedBackgroundView是无效的

/*

 在

 UITableViewCellSelectionStyleBlue, // 在iOS6之前,选中背景色变蓝,iOS7之后,和gray相同




 UITableViewCellSelectionStyleGray, // 默认样式

 UITableViewCellSelectionStyleDefault

NS_ENUM_AVAILABLE_IOS(7_0) // 默认样式

 3种样式下,selectedBackgroundView他的优先级高于selectionStyle

*/

UIView

*cellsBg = [[UIView alloc] init];

cellsBg.backgroundColor = [UIColor redColor];

cell.selectedBackgroundView = cellsBg;

/* 
      contentView = 0x7f96fb553470:

cell.textLabel.superview = 0x7f96fb553470

==0x7f96fb553470

==0x7f96fb553470

*/

 cell上面的子控件不是直接添加在cell上,而是添加到contentView上

NSLog(@"contentView = %p: cell.textLabel.superview =

%p==%p==%p", cell.contentView, cell.textLabel.superview,
cell.imageView.superview, cell.detailTextLabel.superview);

}

  • 注意:cell上面的子控件要添加到cell.contentView上

常见代理/易错属性

  • 前两者易错

选中某一行时候调用

  • (void)tableView:(UITableView *)tableView
    didSelectRowAtIndexPath:(NSIndexPath *)indexPath

{

NSLog(@"%s, row = %ld",__FUNCTION__, indexPath.row);

}

取消选中某一行时候调用

  • (void)tableView:(UITableView *)tableView
    didDeselectRowAtIndexPath:(NSIndexPath *)indexPath

{

NSLog(@"%s, row = %ld",__FUNCTION__, indexPath.row);

}

根据indexPath返回对应的高度

(CGFloat)tableView:(UITableView *)tableView
heightForRowAtIndexPath:(NSIndexPath *)indexPath

{

if (indexPath.row == 0) {

return 222;

}

return 88;

}

  • 分割线样式的枚举容易写错

warning 在设置分割线样式为None的时候如果无效,先看你是否搞混了下列2个枚举

/*

UITableViewCellSeparatorStyleNone

UITableViewCellSelectionStyleNone

*/

self.tableView.separatorStyle =
UITableViewCellSeparatorStyleNone;

cellWithTableView:

cell获取封装

  • 将获取cell的代码全部copy到cell内部,发现需要用到变量tableView,传入即可
  • 传统封装
  • (instancetype)cellWithTableView:(UITableView
    *)tableView

{

 1.确定重用标识

static NSString *ID = nil;

if (ID == nil)
{

ID = [NSString stringWithFormat:@"%@ID",
NSStringFromClass(self)];

}

 2.从缓存池中取

XMGCarCell *cell = [tableView dequeueReusableCellWithIdentifier:ID];

 3.如果缓存池中没有,就自己创建

if (!cell) {

cell = [[XMGCarCell alloc] initWithStyle:UITableViewCellStyleDefault
reuseIdentifier:ID];

}

return cell;

}

  • 注册封装
  • (instancetype)cellWithTableView:(UITableView
    *)tableView

{

 确定重用标示(规范的写法)

static NSString *ID = nil;

if (ID == nil)
{

ID = [NSString stringWithFormat:@"%@ID",
NSStringFromClass(self)];

}



 如果没有注册过,就注册1次

static BOOL isReged = NO;

if (isReged == NO)
{

[tableView registerClass:self
forCellReuseIdentifier:ID];

isReged = YES;

NSLog(@"%s, line = %d",FUNCTION,LINE);

}





 1.缓存池

 这里应该是XMGCarCell *cell,写成id 的原因是为了拖个代码块

id cell = [tableView
dequeueReusableCellWithIdentifier:ID];

return cell;

}

数据覆盖封装

  • 为cell添加模型属性

@property (nonatomic, strong) XMGCar car; /*< carModel */

  • 重写模型set方法完成内部赋值
  • (void)setCar:(XMGCar *)car

{

_car =

car;

self.imageView.image = car.icon;

self.textLabel.text = car.name;

}

封装后创建cell的数据源方法只需要3步

  • 获取cell

  • 覆盖数据

  • 返回cell

  • (UITableViewCell
    *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath
    *)indexPath {

    3.1 获取cell

XMGCarCell *cell = [XMGCarCell cellWithTableView:tableView];

 3.2 数据覆盖

 这样写的好处就是,将cell覆盖数据一行完成,具体覆盖了哪些数据,进入cell内部看

XMGGroup

*g = self.groups[indexPath.section];

XMGCar

*car = g.cars[indexPath.row];

cell.car

= car;

 3.3.返回cell

return cell;

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,874评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,102评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,676评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,911评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,937评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,935评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,860评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,660评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,113评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,363评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,506评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,238评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,861评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,486评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,674评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,513评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,426评论 2 352

推荐阅读更多精彩内容