协议扩展中的变异函数,其中 Self 是 UIViewController

Mutating Function in Protocol Extension Where Self is UIViewController

我已经编写了一个协议和相应的扩展,它利用一个简单的 StringStack"<origin>@<destination>" 形式的命名约定在我的 Swift 中的嵌套视图之间执行 segues应用。我是 swift 的新手,所以如果有更好的方法来实现跨多个视图的前进和后退按钮,我愿意接受。但是,由于我在下面详述的原因,以下代码存在问题。

struct StringStack {
    var stack: Array = [String]()

    mutating func push(str: String)->Void {
        self.stack.append(str)
    }

    mutating func pop()->String {
        self.stack.popLast()
    }
}

protocol SegueForwardAndBackward {
    var thisIdentifier: String {get set}
    var segueStack: StringStack {get set}
}

extension SegueForwardAndBackward where Self: UIViewController {

    mutating func performForwardSegue(nextIdentifier: String) {
        let identifier: String = thisIdentifier + "@" + nextIdentifier
        segueStack.push(identifier)
        performSegueWithIdentifier(identifier, sender: self)
    }

    mutating func performBackwardSegue() {
        let divided = segueStack.pop().componentsSeparatedByString("@")
        // reverses destination and origin of identifier
        let segueIdentifier: String = divided[1] + "@" + divided[0]
        performSegueWithIdentifier(segueIdentifier, sender: self)
    }

    func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
        if var destinationViewController: SegueForwardAndBackward = segue.destinationViewController as? SegueForwardAndBackward {
            destinationViewController.segueStack = segueStack
        }
    }
}

我最初的问题是协议扩展中不允许使用变异函数,详见 bug report。该报告中给出了一个建议的解决方法,但是以下导致错误声称 'SegueForwardAndBackward' is not a subtype of 'UIViewController'.

class A: UIViewController, SegueForwardAndBackward {

    mutating func testFunc() {
        var p: SegueForwardAndBackward = self
        p.performBackwardSegue()
    }
}

我确实遇到了一个潜在的解决方案,我可以使 SegueForwardAndBackward 成为 class 协议,详见 。但是,实施 class 协议会导致段错误,我认为这是因为我将我的扩展声明为 where Self: UIViewController,不幸的是我不知道如何解决这个问题。

目前,我对 Swift 中的另一个错误感到非常沮丧,因此,如果能帮助我改进我的方法或破译这些错误,我将不胜感激。

提前感谢您的回复!

这个 似乎解决了嵌套前进和后退按钮的替代解决方案,因此在没有任何其他响应的情况下,这就是我将要做的。但是,我提出的问题在 swift.

中仍然是一个非常奇怪的错误