ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
AJ分享,必须精品 先看效果图 ![这里写图片描述](https://box.kancloud.cn/2016-03-16_56e8faebddd16.jpg "") ![这里写图片描述](https://box.kancloud.cn/2016-03-16_56e8faec02e38.jpg "") ![这里写图片描述](https://box.kancloud.cn/2016-03-16_56e8faec1e5df.jpg "") ### 源代码 # NYViewController的代码 ~~~ #import "NYViewController.h" #import "NYHero.h" @interface NYViewController () <UITableViewDataSource,UITableViewDelegate> @property (strong,nonatomic) UITableView *tableView; @property (strong,nonatomic) NSArray *heros; @end @implementation NYViewController -(NSArray *)heros { if (_heros == nil)_heros = [NYHero heros]; return _heros; } /**懒加载tableView */ -(UITableView *)tableView { if (_tableView == nil) { //表格控件在创建时必须指定样式 // UITableViewStylePlain 平板格式 // UITableViewStyleGrouped分组格式 _tableView = [[UITableView alloc] initWithFrame:self.view.bounds style:UITableViewStylePlain]; //添加数据源->加入协议 _tableView.dataSource = self; /*设置行高方法有两种,代理方法的优先级比setRowHeight的优先级高。 应用场景,很多应用程序,每一行高度是不一样的,例如:新浪微博 */ // _tableView.rowHeight = 80;//第一种 _tableView.delegate = self;//第二种,要设置代理,-》协议 -》实现代理方法 [self.view addSubview:_tableView]; } return _tableView; } - (void)viewDidLoad { [super viewDidLoad]; [self tableView]; } #pragma mark - 数据源方法 /**每个分组中的数据总数*/ -(NSInteger) tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section { return self.heros.count; } /**告诉表格,每个单元格的明细*/ -(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath { /* UITableViewCellStyleDefault 默认类型 标题+可选图像 UITableViewCellStyleValue1 标题+明细+图像 UITableViewCellStyleValue2 不显示图像,标题+明细 UITableViewCellStyleSubtitle 标题+明细+图像 */ NSLog(@"表格行明细 %d",indexPath.row); //static 静态变量,能够保证系统为变量在内存中只分配一次内存空间 //静态变量,一旦创建,就不会被释放,只有在应用程序在销毁是,才会释放。 static NSString *ID = @"cell"; //1,去缓存池查找可重复用得单元格 UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; //2,如果没找到 if (cell == nil) { // NSLog(@"实例化单元格"); cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID]; cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator; } //设置单元格内容 // 取出英雄对象 NYHero *hero = self.heros[indexPath.row]; //设置标题 :英雄名字 cell.textLabel.text = hero.name; //设置详细内容: 英雄描述 cell.detailTextLabel.text = hero.intro; //设置英雄图标 cell.imageView.image = [UIImage imageNamed:hero.icon]; // 设置右边的箭头 // 1> UITableViewCellAccessoryDisclosureIndicator 箭头,可以"提示"用户,当前行是可以点击的,通常选中行,会跳到新的页面 // 2> UITableViewCellAccessoryCheckmark 对号,通常提示用户该行数据设置完毕,使用的比较少 // 3> UITableViewCellAccessoryDetailButton 按钮,通常点击按钮可以做独立的操作,例如alertView // 点击按钮并不会选中行 // 4> UITableViewCellAccessoryDetailDisclosureButton 按钮+箭头,各自操作 // cell.accessoryType = UITableViewCellAccessoryDetailDisclosureButton; // 指定右侧的自定义视图 /** 通常accessoryType提供的类型不能满足时,才会使用自定义控件 但是需要自行添加监听方法,通常用在自定义cell,不要写在视图控制器中!!! 自定义控件的事件触发,同样不会影响表格行的选中! */ // UISwitch *switcher = [[UISwitch alloc] init]; // //添加监听方法 // [switcher addTarget:self action:@selector(switchChanged:) forControlEvents:UIControlEventValueChanged]; // // cell.accessoryView = switcher; return cell; } -(void)switchChanged:(UISwitch *) sender { NSLog(@"%s %@", __func__, sender); } #pragma mark - 实现代理方法 (行高设置) /**设置行高,比setRowHeight优先级高*/ -(CGFloat) tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { return 70; } // 取消选中某一行,极少用,极容易出错! // didDeselectRowAtIndexPath // didSelectRowAtIndexPath - (void)tableView:(UITableView *)tableView didDeselectRowAtIndexPath:(NSIndexPath *)indexPath { NSLog(@"%s %@", __func__, indexPath); } /**选中了某一行*/ -(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath { NSLog(@"%s %@",__func__, indexPath); } /** accessoryType为按钮时,UITableViewCellAccessoryDetailButton点击右侧按钮的监听方法 此方法不会触发行选中,跟行选中各自独立 只是为accessoryType服务,对自定义控件不响应 */ - (void)tableView:(UITableView *)tableView accessoryButtonTappedForRowWithIndexPath:(NSIndexPath *)indexPath { NSLog(@"%s %@", __func__, indexPath); } @end ~~~ ### 模型的代码的代码 ~~~ #import <Foundation/Foundation.h> @interface NYHero : NSObject @property (nonatomic, copy) NSString *name; @property (nonatomic, copy) NSString *icon; @property (nonatomic, copy) NSString *intro; -(instancetype) initWithDict:(NSDictionary *)dict; +(instancetype) heroWithDict:(NSDictionary *)dict; +(NSArray *) heros; @end ~~~ m实现文件 ~~~ // // NYHero.m // 06 - lol英雄联盟 // // Created by apple on 15-3-28. // Copyright (c) 2015年 znycat. All rights reserved. // #import "NYHero.h" @implementation NYHero -(instancetype)initWithDict:(NSDictionary *)dict { self = [super init]; if (self) { [self setValuesForKeysWithDictionary:dict]; } return self; } +(instancetype)heroWithDict:(NSDictionary *)dict { return [[self alloc] initWithDict:dict]; } +(NSArray *)heros { NSArray *array = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle]pathForResource:@"heros.plist" ofType:nil]]; NSMutableArray *arrayM = [NSMutableArray array]; for (NSDictionary *dict in array) { // 这才是应该写的 [arrayM addObject:[self heroWithDict:dict]]; } return arrayM; } @end ~~~ ### ——关于字典模型的类方法 AJ今天犯二了,字典初始化方法中又一个竟然这么写了,不多说了,大家看看引以为见吧 ~~~ @implementation NYHero -(instancetype)initWithDict:(NSDictionary *)dict { self = [super init]; if (self) { [self setValuesForKeysWithDictionary:dict]; } return self; } +(instancetype)heroWithDict:(NSDictionary *)dict { return [[self alloc] initWithDict:dict]; } +(NSArray *)heros { NSArray *array = [NSArray arrayWithContentsOfFile:[[NSBundle mainBundle]pathForResource:@"heros.plist" ofType:nil]]; NSMutableArray *arrayM = [NSMutableArray array]; for (NSDictionary *dict in array) { //这里我原来是这么写的(注释的是错误的) //arrayM =[self heroWithDict:dict]; // 这才是应该写的 [arrayM addObject:[self heroWithDict:dict]]; } ~~~ 这二犯的,关键用debug你看的话,他会有内容,但是因为名称差不太多,应该是heros返回值是一堆对象组成的数组,结果你NSLog就发现,他丫的就返回一个地址值,很奇怪,开始我还以为是xcode大姨妈来了呢。。。事实证明,代码里面无秘密啊。 ——————话说这好像是AJ写的代码中注释最少的耶。。。 ### 代理模式阶段性小结 监听控件的某些事件 使用代理模式,是为了在程序直接“解耦”。 表格可以显示非常丰富的数据,为了达到这一结果,设置表格的“数据源”。 @required 必须实现的方法。 @optional 可选的实现方法->不强求实现->如果实现了能得到特殊的效果,如果不实现,也不影响程序的正常运行——能够增加控件的灵活度。 代理阶段性小结:(怎么用) 1,遵守协议,预先定义好方法,不实现,具体的实现工作由代理负责。 <控件的名字+DataSource> 定义的与数据有关的方法。 <控件的名字+Delegate> 定义的与事件有关的方法(通常用来监听控件事件)。 2,代理方法: 1> 方法名以控件名称开头(没有类前缀) -> 方便程序员书写的时候,快速找到需要的协议方法。 2> 第一个参数是自己 -> 意味着在协议方法中,可以直接访问对象的属性,或调用方法。 3> 代理方法的返回值 -> 控制器向控件(委托)发送数据 。 ### cell——UITableViewCell的注意点 缓存池的运用(老板本,后面会有新的东西更新,但是需要了解,看别人程序时候别不认识了——还有方便理解) ~~~ //1,去缓存池查找可重复用得单元格 UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:ID]; //2,如果没找到 if (cell == nil) { // NSLog(@"实例化单元格"); cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID]; cell.accessoryType = UITableViewCellAccessoryDisclosureIndicator; } //3设置单元格内容 //4return cell ~~~ 初始化的时候需要指定cell的样式 UITableViewCellStyleDefault 默认类型 标题+可选图像 UITableViewCellStyleValue1 标题+明细+图像 UITableViewCellStyleValue2 不显示图像,标题+明细 UITableViewCellStyleSubtitle 标题+明细+图像 用法: ~~~ cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:ID]; ~~~ ### 指定cell右侧的视图 设置右边的箭头 1> UITableViewCellAccessoryDisclosureIndicator 箭头,可以”提示”用户,当前行是可以点击的,通常选中行,会跳到新的页面 2> UITableViewCellAccessoryCheckmark 对号,通常提示用户该行数据设置完毕,使用的比较少 3> UITableViewCellAccessoryDetailButton 按钮,通常点击按钮可以做独立的操作,例如alertView 点击按钮并不会选中行 4> UITableViewCellAccessoryDetailDisclosureButton 按钮+箭头,各自操作 cell.accessoryType = UITableViewCellAccessoryDetailDisclosureButton; 指定右侧的自定义视图(例:选择按钮) 通常accessoryType提供的类型不能满足时,才会使用自定义控件 但是需要自行添加监听方法,通常用在自定义cell,不要写在视图控制器中!!! 自定义控件的事件触发,同样不会影响表格行的选中! ~~~ UISwitch *switcher = [[UISwitch alloc] init]; //添加监听方法 [switcher addTarget:self action:@selector(switchChanged:) forControlEvents:UIControlEventValueChanged]; cell.accessoryView = switcher; ~~~