【问题标题】:Is a protocol either a reference or value type in Swift?Swift 中的协议是引用类型还是值类型?
【发布时间】:2019-02-07 21:02:54
【问题描述】:

我问这个问题的原因是因为我正在阅读使用委托的教程。根据我从其他在线教程/文章中阅读的内容,据我所知,该特定教程并未创建保留周期。我还使用 Instruments(内存泄漏和僵尸)对其进行了测试,以确保没有内存泄漏。

我试图弄清楚一个类是否只是遵循协议,这会创建引用吗?

我认为不会,但我想确定。

这是创建委托成员的协议和类:

import Foundation
import CoreBluetooth

protocol TransferServiceScannerDelegateProtocol: NSObjectProtocol {
    func didStartScan()
    func didStopScan()
    func didTransferData(data: NSData?)
}

class TransferServiceScanner: NSObject, CBCentralManagerDelegate, CBPeripheralDelegate {
    var centralManager: CBCentralManager!
    var discoveredPeripheral: CBPeripheral?
    var data = NSMutableData()
    weak var delegate: TransferServiceScannerDelegateProtocol?

    init(delegate: TransferServiceScannerDelegateProtocol?) {
        super.init()
        centralManager = CBCentralManager(delegate: self, queue: nil)
        self.delegate = delegate
    }


    //start of cbCentralDelegate method
    func centralManagerDidUpdateState(_ central: CBCentralManager) {
        switch central.state {
        case .poweredOn:
            print("Central is powered on...")
            break
        case .poweredOff:
            print("Central is powered off...")
            break
        default:
            print("Central manager changed state \(central.state)")
            break
        }
    }
    //end of cbCentralDelegate method

}

这是一个视图控制器,它符合协议,但也有上面提到的类型类的属性。我还不确定为什么这个视图控制器符合协议,但我认为这不会增加引用计数:

import UIKit

class CentralViewController: UIViewController, TransferServiceScannerDelegateProtocol {
    @IBOutlet var heartImage: UIImageView!
    @IBOutlet var scanButton: CustomButton!
    @IBOutlet var textView: UITextView!
    var transferServiceScanner: TransferServiceScanner!

    // MARK: TransferServiceScannerDelegateProtocol methods
    func didStartScan() {
        //
    }

    func didStopScan() {
        //
    }

    func didTransferData(data: NSData?) {
        //
    }
    //end of TransferServiceScannerDelegateProtocol methods


    override func viewDidLoad() {
        super.viewDidLoad()
        transferServiceScanner = TransferServiceScanner.init(delegate: self)
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
    }


    /*
    // MARK: - Navigation

    // In a storyboard-based application, you will often want to do a little preparation before navigation
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        // Get the new view controller using segue.destinationViewController.
        // Pass the selected object to the new view controller.
    }
    */

}

【问题讨论】:

  • I'm not sure yet as to why this view controller conforms to the protocol 不清楚你的意思。
  • 好吧,到目前为止,我们已经将协议函数放在了这个视图控制器中,但是我相信由于 TransferServiceScanner 类已经有一个这种协议类型的变量,所以我认为没有必要。我认为没有必要,因为视图控制器创建了一个 TransferServiceScanner 类型的属性。
  • I believe that since the TransferServiceScanner class already has a variable of this protocol type then I don't think its necessary. 如果类型符合协议,则只能将类型用作协议,并且必须声明一致性,无论该类型是否已实现必要的方法。如果你不声明视图控制器符合,那么你不能把它当作委托,因为你不能把它交给服务扫描器,因为它需要一个符合协议的对象。
  • 你是对的,它在移除视图控制器中的一致性后崩溃了。

标签: ios swift delegation


【解决方案1】:

我认为这不应该增加引用计数:

var transferServiceScanner: TransferServiceScanner 将引用计数增加到 1,因为所有引用都是强引用,如果它们没有被声明为弱或其他。

将委托变量存储为 weak 可确保强引用不会双向使用,因此 ARC 可以取消初始化它们。

我试图弄清楚一个类是否只是遵循协议,这会创建引用吗?

一个类总是一个引用类型,无论你是通过协议还是直接引用它。因此,分配一个带有引用类型(类)的协议并不会复制类对象,而是给出另一个对该对象的引用并增加 ARC 查看的引用计数。

protocol TransferServiceScannerDelegateProtocol: NSObjectProtocol {

您确保只有一个类可以实现该协议,这就是为什么您可以声明weak var delegate: TransferServiceScannerDelegateProtocol,因为只有类可以实现NSObjectProtocolNSObject & co。

无需声明协议类,无论是结构还是类,都可以实现协议。但只有当您将协议限制为仅类时,您才能像使用类一样使用协议,使用 weak 之类的东西。

【讨论】:

  • 更简洁的语法含义完全相同的是protocol TransferServiceScannerDelegateProtocol: class {
猜你喜欢
  • 2010-12-04
  • 1970-01-01
  • 1970-01-01
  • 2013-07-14
  • 2013-01-11
  • 1970-01-01
  • 2018-05-21
  • 2010-11-07
  • 1970-01-01
相关资源
最近更新 更多