协议在Swift中是引用类型还是值类型?

时间:2018-09-02 16:38:50

标签: ios swift delegation

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

我正在尝试确定某个类是否仅符合协议,这会创建引用吗?

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

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

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.
    }
    */

}

1 个答案:

答案 0 :(得分:1)

  

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

var transferServiceScanner: TransferServiceScanner将引用计数增加到一个,因为如果未将所有引用声明为弱引用或其他引用,则所有引用都是强引用。

将委托变量存储为weak可确保强引用不会同时出现,因此ARC可以将其初始化。

  

我正在尝试确定某个类是否仅符合协议,这会创建引用吗?

无论通过协议还是直接引用类,类始终是引用类型。因此,分配在其后面具有引用类型(类)的协议不会复制该类对象,而是您要提供对该对象的另一个引用并增加ARC所查看的引用计数。

使用

protocol TransferServiceScannerDelegateProtocol: NSObjectProtocol {

您要确保只有一个类才能实现协议,这就是为什么可以声明weak var delegate: TransferServiceScannerDelegateProtocol的原因,因为只有类可以与NSObjectProtocol和co一起实现NSObject

在不声明仅协议类的情况下,结构或类都可以实现协议。但是只有将协议限制为仅类,才能像使用类一样使用协议,并使用weak之类的东西。