天天看点

swift之mutating关键字【转】

在swift中,包含三种类型(type): structure,enumeration,class

其中structure和enumeration是值类型(value type),class是引用类型(reference type)

但是与objective-c不同的是,structure和enumeration也可以拥有方法(method),其中方法可以为实例方法(instance method),也可以为类方法(type method),实例方法是和类型的一个实例绑定的。

在swift官方教程中有这样一句话:

[plain] view plaincopy在CODE上查看代码片派生到我的代码片

“Structures and enumerations are value types. By default, the properties of a value type

cannot be modified from within its instance methods.”

摘录来自: Apple Inc. “The Swift Programming Language”。 iBooks. https://itunes.apple.com/WebObjects/MZStore.woa/wa/viewBook?id=881256329

大致意思就是说,虽然结构体和枚举可以定义自己的方法,但是默认情况下,实例方法中是不可以修改值类型的属性。

举个简单的例子,假如定义一个点结构体,该结构体有一个修改点位置的实例方法:

[plain] view plaincopy在CODE上查看代码片派生到我的代码片

struct Point {  
    var x = , y =   

    func moveXBy(x:Int,yBy y:Int) {  
        self.x += x  
        // Cannot invoke '+=' with an argument list of type '(Int, Int)'  
        self.y += y  
        // Cannot invoke '+=' with an argument list of type '(Int, Int)'  
    }  
}  
           

编译器抛出错误,说明确实不能在实例方法中修改属性值。

为了能够在实例方法中修改属性值,可以在方法定义前添加关键字mutating

struct Point {  
    var x = , y =   

    mutating func moveXBy(x:Int,yBy y:Int) {  
        self.x += x  
        self.y += y  
    }  
}  

var p = Point(x: , y: )  

p.moveXBy(, yBy: )  
           

另外,在值类型的实例方法中,也可以直接修改self属性值。

enum TriStateSwitch {  
    case Off, Low, High  
    mutating func next() {  
        switch self {  
        case Off:  
            self = Low  
        case Low:  
            self = High  
        case High:  
            self = Off  
        }  
    }  
}  
var ovenLight = TriStateSwitch.Low  
ovenLight.next()  
// ovenLight is now equal to .High  
ovenLight.next()  
// ovenLight is now equal to .Off”  
           

TriStateSwitch枚举定义了一个三个状态的开关,在next实例方法中动态改变self属性的值。

当然,在引用类型中(即class)中的方法默认情况下就可以修改属性值,不存在以上问题。