iOS——记一次block的循环引用

最近在做的小项目出现了一个问题。当presentViewController时内存上升(在debug navigator下观察),dissmissViewController时竟然没有下降。我第一反应就是哪里出现循环引用了。于是我手工遍历一遍代码,还好代码量不是太多,很快就找到问题的根源。

下文的block其实就是swift中的closure闭包,两者非常想想。

问题描述:

有时候为了减少重复代码,我们会将重复的代码抽取出去放在一个函数里面,然后根据不同参数来调用该函数。

    func editAccountBook(item:AccountBookBtn?, indexPath:NSIndexPath, sureBlock:(String, String)->Void){
        customAlertView.title = item?.btnTitle ?? ""
        customAlertView.initChooseImage = item?.backgrountImageName ?? "book_cover_0"
        customAlertView.cancelBlock = {[weak self] in
            if let strongSelf = self{
                strongSelf.customAlertView.removeFromSuperview()
            }
        }
        customAlertView.sureBlock = sureBlock
        UIApplication.sharedApplication().keyWindow?.addSubview(self.customAlertView)
    }

比如上面这个函数,目光集中在倒数第二行的sureBlock,这个sureBlock是外部传进来的。我们知道Block的原理就是一个结构体,像函数传参这种一般都是传sureBlock的结构体指针。在使用属性的时候,swift允许我们不加self,所以customAlerView是被self引用到的。而只要这个sureBlock里面有对self的强引用,将sureBlock赋值给customAlerView.sureBlock的话就会造成循环易用。所以说,抽代码要小心呐。

再看看editAccountBook这个函数被调用的地方,一开始我是这么写的

    let block:(String,String)->Void = {(title, imageName) in
        //建一个数据库
        let currentTime = Int(NSDate().timeIntervalSince1970)
        let dbName = customAccountName + "\(currentTime)" + ".db"
        let item = AccountBookBtn(title: title, count: "0笔", image: imageName, flag: false, dbName: dbName)
        //插入账本
        self.mainVCModel.addBookItemByAppend(item)
        self.mainView.accountBookBtnView.insertItemsAtIndexPaths([indexPath])
        //退出alertview
        self.customAlertView.removeFromSuperview()
    }
    editAccountBook(nil, indexPath: indexPath, sureBlock: block)

当时觉得这个block是临时生成的,里面虽然引用到self,应该也没什么关系。殊不知这个block在传给editAccountBook这个函数的时候就间接地被self引用到了,所以就造成了循环引用。所以用block真是要特别小心。特别是一些自己写的函数。

解决办法

解决办法很简单,在block的开头加一些list就好。如下

    editAccountBook(nil, indexPath: indexPath){[weak self](title, imageName) in
        if let strongSelf = self{
            //建一个数据库
            let currentTime = Int(NSDate().timeIntervalSince1970)
            let dbName = customAccountName + "\(currentTime)" + ".db"
            let item = AccountBookBtn(title: title, count: "0笔", image: imageName, flag: false, dbName: dbName)
            //插入账本
            strongSelf.mainVCModel.addBookItemByAppend(item)
            strongSelf.mainView.accountBookBtnView.insertItemsAtIndexPaths([indexPath])
            //退出alertview
            strongSelf.customAlertView.removeFromSuperview()
        }
    }

这里用了swift的省略写法,当函数最后一个参数是block时,可以将整个block移到函数的末尾。可以看到这里使用了[weak self],这一句会将selfwrap成一个optional,所以在使用的时候得unwrap。其中的if let strongSelf = self就是unwrap。看到这里,用Objc开发的同学应该很熟悉了,在ARC下block的写法和swift里的思想都是一模一样的,格式有点不同罢了。

自己写的总结,方便以后查看。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 《Objective-C高级编程》这本书就讲了三个东西:自动引用计数、block、GCD,偏向于从原理上对这些内容...
    WeiHing阅读 13,325评论 10 69
  • 此人画渣 第一次画萌物 第一次使用马克笔 学习之余的画 一做完作业立刻提起画笔,以此释放我的压力 第一次画框 对于...
    萌萌哒喵喵酱阅读 2,744评论 3 1
  • 很多人工作了好些年,一直得不到升迁,看着资历比自己低的已经位居高位了,心中难免有怨气,天天在抱怨:老子拼死拼活干得...
    趣昂阅读 2,944评论 0 1
  • 2016年国庆,90后的我裸辞了,带着两身衣服回到家。做短暂的啃老族,中旬奔赴另一个城市从事其他的行业岗位。 现在...
    木筱菱阅读 3,582评论 0 0