控制器加载内容时,用户点击返回,控制器销毁时网络请求处理
/** 获取数据*/
- (void)loadMenuData
{
NSMutableDictionary *params = [NSMutableDictionary dictionary];
params[@"a"] = @"tag_recommend";
params[@"action"] = @"sub";
params[@"c"] = @"topic";
// 加载网络时需要显示加载动画
[SVProgressHUD show];
_manager =[AFHTTPSessionManager manager];
// 不设置弱引用的话控制器释放不了 一定要等到block执行完毕后才会释放,block默认是对变量强引用的,所以需要创建一个弱引用变量
__weak typeof(self) weakSelf = self;
[_manager GET:commonUrl parameters:params progress:nil success:^(NSURLSessionDataTask * _Nonnull task, id _Nullable responseObject) {
// 模拟网络延迟
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
[SVProgressHUD dismiss];
weakSelf.dataArr = [NSArray yy_modelArrayWithClass:[VJMenu class] json:responseObject];
[weakSelf.tableView reloadData];
});
} failure:^(NSURLSessionDataTask * _Nullable task, NSError * _Nonnull error) {
//关闭网络请求时也会来到failure这 所以需要判断是不是因为控制器返回关闭网络请求导致的
if (error.code == NSURLErrorCancelled) return ;
[SVProgressHUD dismiss];
[SVProgressHUD showErrorWithStatus:@"请求失败"];
}];
}
- (void)viewWillDisappear:(BOOL)animated
{
[SVProgressHUD dismiss];
//关闭当前session的网络请求
[_manager invalidateSessionCancelingTasks:YES];
}
监听keyboard尺寸修改发送的信号移动输入框控件
/** 初始化*/
- (void)setUpBase
{
self.navigationItem.title = @"评论";
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardChange:) name:UIKeyboardWillChangeFrameNotification object:nil];
}
/** 键盘更改frame信号事件*/
- (void)keyboardChange:(NSNotification*)note
{
CGFloat diffHeight = [UIScreen mainScreen].bounds.size.height - [note.userInfo[UIKeyboardFrameEndUserInfoKey] CGRectValue].origin.y;
_buttonConst.constant = diffHeight;
CGFloat duration = [note.userInfo[UIKeyboardAnimationDurationUserInfoKey] floatValue];
[UIView animateWithDuration:duration animations:^{
[self.view layoutIfNeeded];
}];
}
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{
[self.view endEditing:YES];
}
NSNotifcationCenter使用
- 实例化
- [NSNotifcationCenter defaultCenter]
- 添加观察者(监听信号)
- addObserver:xxx selector:$sel name:xxx object:xxx
- $sel会接受到一个NSNotification类型参数通过userinfo属性可以拿到发送该信号时传递的参数。
- addObserver:xxx selector:$sel name:xxx object:xxx
- 发送信号
- postNotifactionName:xxx object:xxx userinfo:xxx
- 移除观察者(移除监听)
- removeObserver:xxx name:xxx object:xxx
注意:
- 监听信号 移除信号的object必须和发送信号的object一样,如果发送信号的object是nil则监听信号和移除信号的object也必须是nil.
- 监听信号 移除信号的object如果是nil的话 则可以监听或者移除所有的信号(无论发送信号的object是否指定对象)
UITableViewSectionHeader自定义及浮动顶部
style为Plain的tableview设置了section的header的话,该header滚动时会浮动到顶部,自定义该header需要实现UITableViewDelegate代理的viewForHeaderInSection方法 返回一个自定义的view
注意:自定义section的headerView时需要实现heightForHeaderInSection代理方法,设置sectionHeaderView的高度,否则不会显示sectionHeaderView.
TableViewCell根据内容计算高度
//设置单元行的预估高度
self.tableView.estimatedRowHeight = 46;
//设置row的高度为自动计算
self.tableView.rowHeight = UITableViewAutomaticDimension;
//然后自定义一个cell的nib
//设置某个控件的底部与contentView相当
//由于contentView没有约束所以看起来像会有点不正确(因为不会自动调整contentView的尺寸)实际渲染是正确的
由控件的约束限定了单元行的高度后就能自动计算出单元行的高度。
NAarray方法
- 让数组内所有对象统一执行某个方法
[$array makeObjectsPerformSelector:$sel];
//让所有元素执行该sel方法
- 添加另一个数组内的所有元素
[$array addObjectsFromArray:$otherArray];
AFHttpSession取消其他未执行的冲突的网络请求
/**
tasks当前AFHttpSessionManger的所有网络任务
tasks是一个数组
可以调用makeObjectsPerformSelector:cancel
来使所有任务取消执行
*/
[self.manger.tasks makeObjectsPerformSelector:@selector(cancel)];
深拷贝对象
- 深拷贝字符串、字典、数组。
- 深拷贝对象
- 给自定义对象实现NSCopying协议
- 实现copyWithZone方法 返回一个新的对象,当对象调用copy方法时就会返回该方法生成的对象。
//示例:
- (id)copyWithZone:(NSZone *)zone{
//此处必须新生成一个对象!!!
VJTopic *topic = [[self class] yy_modelWithDictionary:[self yy_modelToJSONObject]];
topic.contentF = self.contentF;
return topic;
}
//调用:
VJTopic *copyTopic = [self.topic copy];
关于block对象强引用
记住下面两句话
- block内如果【使用了外部声明的强引用】访问某个对象,则block内部【就会产生一个强引用】引用该对象!
- block内如果【使用了外部声明的弱引用】访问某个对象,则block内部【就会产生一个弱引用】引用该对象!防止循环引用
- block内嵌套block时防止第一个作用域弱引用的对象被释放,需要再创建一个变量(局部强引用)执行该对象,然后让嵌套的作用域对该对象进行强引用
注意:block内产生的引用都是执行对象内存地址值,和变量没关系,千万不要理解为是引用某个变量(错误的),引用的是变量指向的对象的内存。
- 举个例子:
void test(){
VJPerson *p = [VJPerson new];
p.name = @"vijay";
// 创建一个弱引用
__weak typeof(p) weakP = p;
void(^block1)() = ^() {
// 此处引用了外部声明的weakP弱引用 则block1会创建一个弱引用执行对象P内存地址值.
NSLog(@"%@",weakP.name);
// 创建一个变量(当前作用域的强引用变量)指向对象p的内存地址值(该变量会在代码块执行完后销毁)
VJPerson *strongP = weakP;
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
// 使用了外面声明的强引用变量strongP 则该block会创建一个强引用指向对象P内存地址值.
NSLog(@"%@",weakP.name);
NSLog(@"%@",strongP.name);
});
};
// 执行block1后strongP变量被销毁
block1();
// 代码块完全执行完后block2中的还指向着对象P所以对象P还没被释放 等到block2执行完后释放
}
判断控件坐标系
- 同坐标系判断
- CGRectContainsRect(rect1,rect2); //是否包含
- CGRectIntersectsRect(rect1,rect2); //是否交叉
- 不同坐标系判断
- [view1 convertRect:view2.frame fromView:view2];
- 将view2.frame得到的CGRect从view2的坐标系转到view1结果为该CGRect在view1的坐标系
- [view1 convertRect:$rect toView:view2];
- 将Rect从view1的坐标系上改成view2的坐标系
- toView参数为nil时 默认是获取相对UIWindow的坐标系
注意:不要在viewdidload方法中转换坐标系。调用viewdidload时可能某些控件还没加入父控件的subview上