ios开发技巧(二)

控制器加载内容时,用户点击返回,控制器销毁时网络请求处理



/** 获取数据*/
- (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属性可以拿到发送该信号时传递的参数。
  • 发送信号
    • postNotifactionName:xxx object:xxx userinfo:xxx
  • 移除观察者(移除监听)
    • removeObserver:xxx name:xxx object:xxx

注意:

  1. 监听信号 移除信号的object必须和发送信号的object一样,如果发送信号的object是nil则监听信号和移除信号的object也必须是nil.
  1. 监听信号 移除信号的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上

你可能感兴趣的:(ios开发技巧(二))