delegate 为什么应该是 weak 类型而不是strong类型

delegate 为什么应该是 weak 类型而不是strong类型


循环引用
对象 a 创建并引用了对象 b. 对象 b 创建并引用了对象 c. 对象 c 创建并引用了对象 b.
这时候 b c 的引用计数分别是 2 1 。当 a 不再使用 b ,调用 release 释放对 b 的所有权,因为 c 还引用了 b ,所以 b 的引用计数为 1 b 不会被释放。 b 不释放, c 的引用计数就是 1 c 也不会被释放。从此, b c 永远留在内存中。

打断这种循环引用:
如果c引用b的时候是用 weak类型弱引用,那么a释放b时,b的引用计数器变为0,b被释放,c也被释放。


一个例子:

// A.m中某处 

B* b = [B alloc] init]; 

b.delegate = self

[self.view addSubview:b];

[b release];

原因1 

delegate没有必要是strong A负责创建B的,A的生命周期一定比B要长(B存在,A一定也存在;A存在,B不一定存在) 也就是说 B(实例b)存在的时候,A(的实例)一定存在, 所以没有必要用retain将引用计数+1assign足以 

原因2 

退一步假设存在这样的情况:AB先挂掉(A的实例先被release销毁) 那么当然希望是[a relase]后,A中的方法不再被B调用,但是使用retain时候,A还是没有被销毁,so A中的方法仍会被B调用,但是assgin就无此问题 

原因3 

避免循环引用 两个类互相强引用,referance counting永远不会是0,一直会占用内存,如果delegate是弱引用,那delegate释放掉之后,这个类也可以被释放


另一个例子:


一个 UITableViewController 对象 a 通过 retain 获取了 UITableView 对象 b 的所有权,这个 UITableView 对象 b delegate 又是 a 如果这个 delegate retain 方式的,那基本上就没有机会释放这两个对象了。 循环引用而产生的内存泄露也是 Instrument 无法发现的

所以delegate必须设置为weak类型




参考:
iOS 5 ARC完全指南.pdf 
http://blog.csdn.net/diyagoanyhacker/article/details/6591593
http://www.cocoachina.com/ask/questions/show/93387


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以为您提供一个简单的示例。假设您有一个名为`CustomData`的自定义数据类型,它有两个属性:`name`和`age`。 首先,您需要编写一个自定义的Delegate类,该类继承自QItemDelegate,并且需要重写createEditor()、setEditorData()、setModelData()和updateEditorGeometry()方法。 ```python from PyQt5.QtCore import Qt from PyQt5.QtGui import QIntValidator from PyQt5.QtWidgets import QItemDelegate, QLineEdit class CustomDelegate(QItemDelegate): def createEditor(self, parent, option, index): editor = QLineEdit(parent) validator = QIntValidator(0, 100, editor) editor.setValidator(validator) return editor def setEditorData(self, editor, index): value = index.model().data(index, Qt.EditRole) editor.setText(str(value.age)) def setModelData(self, editor, model, index): value = model.data(index, Qt.EditRole) value.age = int(editor.text()) model.setData(index, value, Qt.EditRole) def updateEditorGeometry(self, editor, option, index): editor.setGeometry(option.rect) ``` 然后,您需要在使用QTableView时将该Delegate应用于您的模型。例如: ```python from PyQt5.QtCore import Qt, QAbstractTableModel, QVariant, QModelIndex from PyQt5.QtWidgets import QTableView, QApplication class CustomData: def __init__(self, name, age): self.name = name self.age = age class CustomTableModel(QAbstractTableModel): def __init__(self, data): super().__init__() self._data = data def data(self, index, role=Qt.DisplayRole): if role == Qt.DisplayRole: if index.column() == 0: return self._data[index.row()].name elif index.column() == 1: return self._data[index.row()].age elif role == Qt.EditRole: return self._data[index.row()] def setData(self, index, value, role=Qt.EditRole): if role == Qt.EditRole: self._data[index.row()] = value self.dataChanged.emit(index, index) return True return False def rowCount(self, parent=QModelIndex()): return len(self._data) def columnCount(self, parent=QModelIndex()): return 2 def headerData(self, section, orientation, role=Qt.DisplayRole): if role == Qt.DisplayRole and orientation == Qt.Horizontal: if section == 0: return "Name" elif section == 1: return "Age" return QVariant() if __name__ == '__main__': import sys app = QApplication(sys.argv) data = [CustomData("Tom", 20), CustomData("Jerry", 30), CustomData("Lucy", 25)] model = CustomTableModel(data) view = QTableView() view.setModel(model) delegate = CustomDelegate(view) view.setItemDelegate(delegate) view.show() sys.exit(app.exec_()) ``` 这样,当您在TableView中编辑一行时,Delegate会将其转换为CustomData类型,并将其存储到Model中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值