Swift的代理delegate

在Swift实际开发中,很容易在声明delegate时忽略掉使用weak,这是一个比较严重的问题。现对delegate造成的循环引用的原因及解决办法,还有swift中有关delegate的使用语法进行了一番总结。

关于闭包的循环引用相关问题请看:
Swift闭包循环引用

该文章主要讲解:

  • 使用delegate如何造成的循环引用问题
  • 怎么使用weak解决循环引用问题
  • delegate的一些比较Swift的用法

在这里我模拟一个老师Teacher和学生Student互动的场景,Teacher提问askQuestion StudentStudent回答问题giveAnswer,把问题的答案回调delegateTeacher,最后由Teacher判断答案是否正确judgeIsRightOrNot(answer: Int)

Student.swift文件

protocol StudentDelegate {
    func judgeIsRightOrNot(answer: Int)
}

class Student {
    
    var delegate: StudentDelegate?
    
    // 回答问题
    func giveAnswer() {
        delegate?.judgeIsRightOrNot(answer: 1)
    }
    
    deinit {
        print("deinit---Student")
    }
}

Teacher.swift文件

class Teacher: StudentDelegate {
    
    var student: Student?
    
    init() {
        self.student = Student()
        self.student?.delegate = self
    }
    
    // 提问问题
    func askQuestion() {
        // 学生回答
        student?.giveAnswer()
    }

    // 判断答案是否正确
    func judgeIsRightOrNot(answer: Int) {
        if answer == 1 {
            print("Right")
        } else {
            print("Error")
        }
    }
    
    deinit {
        print("deinit---Teacher")
    }
}

ViewController.swift文件

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
       
        let teacher = Teacher()
        teacher?.askQuestion()
    }
    
    deinit {
        print("deinit---ViewController")
    }
}

ViewControllerTeacher是弱引用,viewDidLoad方法执行完之后就不再对Teacher进行引用,Teacher就应该被释放掉了,还有Student也是。

然而运行后却发现,TeacherStudent两者的deinit方法都没走,也就是说这两个对象都没有被释放掉!

Why???

这就回归到了问题的正题,因为delegate是强引用导致的!
来回顾一下代码:

 class Teacher {
    var student: Student
 }
 self.student?.delegate = self

第一行代码var student: Student说明teacher是对student进行了强引用
然后第二行代码self.student?.delegate = self表明student.delegate又对teacher进行了强引用,使teacher的retainCount + 1

所以当viewController不对teacher引用后,self.student.delegate却还对teacher强引用着,使得teacherretainCount = 1,所以teacher不会释放,student同样也无法释放,所以就导致了两者之间相互循环引用

解决方法

把delegate用weak进行修饰
class Student {
    weak var delegate: StudentDelegate?
}

使用weak之后,虽然teacherstudent还是强引用,但是self.student.delegateteacher变成弱引用了,不会再让teacherretainCount + 1

因此当ViewControllerviewDidLoad执行完毕后,就对teacher不再进行引用,所以teacher的retainCount变成了0后,teacher就被释放了,同样student也就被释放掉了,大家皆大欢喜~

进一步思考一个小问题

如果ViewController中的teacher不是在viewDidLoad中进行声明的,而是一个全局变量呢?当控制器被pop掉后结果会怎样呢,ViewController会被释放吗?

其实我们要搞明白一件事情,就是刚才的teacherstudent没被释放是因为它们两者之间相互引用所造成的,而现在ViewController并没有和其他对象构成循环引用,所以在ViewController被pop掉后就立马被释放了,没有什么影响,除非它也像teacher刚才那样与某个对象形成了循环引用关系,retainCount + 1最终才会不被释放。

而对于teacherstudent来说,分为以下两种情况:

1) 不使用weak,造成循环引用

既然在viewDidLoad执行完也就是teacher的生命周期结束后都没有被释放,那么即便是控制器被pop掉也依然不会被释放

2) 使用weak,没有被循环引用

既然teacher没有被循环引用,所以在它生命周期结束后(因为是全局变量,所以也就是当ViewController被pop掉内存被释放掉后),teacher不再被其引用,因此会被释放,然后student也会被释放。
所以控制台打印的结果是:

deinit---TwoController
deinit---Teacher
deinit---Student



delegate的相关语法补充

最后在补充一下delegate的相关语法,比如刚才,大家实验一下会发现delegate用weak修饰后,编译报错了。。。

image.png

报错原因:

这是因为Swiftprotocol是可以被除了class以外的其他类型遵守的,而对于像structenum这样的类型,本身就不通过引用计数来管理内存,所以也不可能用weak这样的ARC的概念来进行修饰。

解决办法:

1)将protocol声明为Objective-C的,这可以通过在protocol前面加上@objc关键字来达到,Objective-Cprotocol都只有类能实现,因此使用weak来修饰就合理了。
@objc protocol StudentDelegate {
    func judgeIsRightOrNot(answer: Int)
}
2)在protocol声明的名字后边加上class,这可以为编译器显示地指明这个protocol只能由class来实现。
protocol StudentDelegate: class {
    func judgeIsRightOrNot(answer: Int)
}

相比起添加@objc,后一种方法更能表现出问题的实质,同时也避免了过多的不必要的Objective-C兼容。

3)但是在实际开发中,我见很多人在protocol声明的名字后边没有加class,而是加的NSObjectProtocol,我试了一下发现报错了。。
image.png

它说Teacher类没有遵守NSObjectProtocol协议,然后我看了下NSObjectProtocol的介绍,如下:

image.png

我英语比较烂,看的似懂非懂。。。我感觉它的意思是StudentDelegate遵循了它,而Teacher类如果要实现StudentDelegate协议,就要继承自NSObject(而UIViewUIViewController则不需要,因为它们本身就继承自NSObject),我猜的,不知道这么理解对不对,哈哈😁😁。。。如果有知道的大神,敬请指导~

我把Teacher类添加了继承自NSObject,然后把init方法稍微修改了一下,报错消失,编译通过

class Teacher: NSObject, StudentDelegate  {
    override init() {
        super.init()
        student = Student()
        student?.delegate = self
    }
}

代理的可选方法

1)在protocol定义之前加上@objc

原生的Swift protocol里没有可选项,所有定义的方法都是必须实现的,如果我们想要像Objective-C里那样定义可选的接口方法,就需要将接口本身定义为Objective-C的,也就是在protocol定义之前加上@objc

@objc protocol OptionalProtocol {
    @objc optional func optionalMethod() // 可选方法
    func necessaryMethod() // 必须实现的方法
}

2)使用protocol extension

使用@objc修饰的protocol就只能被class实现,这对于structenum类型就无法适用了。另外,实现它的class中的方法也必须被标注为@objc,或者整个类就是继承自NSObject,这对写代码就有了一些限制。
还有另一种选择就是使用protocol extension,我们可以在声明一个protocol之后再用extension的方式给出部分方法的默认实现,这样这些方法在实际的类中就是可选实现的了。

protocol OptionalProtocol {
    func optionalMethod() // 可选的方法
    func necessaryMethod() // 必须实现的方法
}

extension OptionalProtocol {
    // 在扩展中给出了默认实现的方法,在实际类中就是可选的了
    func optionalMethod() {
        print("默认实现")
    }
}



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

推荐阅读更多精彩内容

  • 无论OC中的Block还是Swift中的闭包Closure,经常因为使用不当从而造成循环引用从而导致内存泄漏,如何...
    Tony_Yang阅读 4,662评论 2 21
  • Swift 介绍 简介 Swift 语言由苹果公司在 2014 年推出,用来撰写 OS X 和 iOS 应用程序 ...
    大L君阅读 3,168评论 3 25
  • 1.objective-c常见面试题:1、**OC **语言的基本特点OC 语言是 C 语言的一个超集,只是在 C...
    LZM轮回阅读 957评论 0 3
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,119评论 29 470
  • 多少人在生命中与自己越走越远。 也许会有人觉得我的题目很奇怪,想说我们 自己不就是自己么?怎么能自己跟自己越走越远...
    昵称都被占了阅读 340评论 0 1