通过这个简单的类,我得到了编译器警告

试图在自己的setter/getter中修改/访问x

当我这样使用它时:

var p: point = Point()
p.x = 12

我得到一个EXC_BAD_ACCESS。我怎么能做到这一点没有显式支持变量?

class Point {

    var x: Int {
        set {
            x = newValue * 2 //Error
        }
        get {
            return x / 2 //Error
        }
    }
    // ...
}

当前回答

我不知道这是不是一个好的练习,但是你可以这样做:

class test_ancestor {
    var prop: Int = 0
}


class test: test_ancestor {
    override var prop: Int {
        get {
            return super.prop // reaching ancestor prop
        }
        set {
            super.prop = newValue * 2
        }
    }
}

var test_instance = test()
test_instance.prop = 10

print(test_instance.prop) // 20

阅读更多

其他回答

这里有一个理论上的答案。可以在这里找到

{get set}属性不能是常量存储属性。它应该是一个计算属性,get和set都应该被实现。

你用x递归地定义x,就好像有人问你多大了?你回答"我的年龄是我的两倍"这毫无意义。

你必须说我的年龄是约翰的两倍或者其他变量,除了你。

计算变量总是依赖于另一个变量。


经验法则是永远不要从getter (get)中访问属性本身。因为这会引发另一个get,而这个get又会引发另一个get…甚至不要打印出来。因为打印也需要在打印之前“获取”值!

struct Person{
    var name: String{
        get{
            print(name) // DON'T do this!!!!
            return "as"
        }
        set{
        }
    }
}

let p1 = Person()

因为这将给出以下警告:

试图从它自己的getter中访问'name'。

错误看起来像这样模糊:

作为替代,你可能想要使用didSet。使用didSet,你会得到一个值,这个值是之前设置的,刚刚设置的。更多信息请看这个答案。

试试这个:

var x:Int!

var xTimesTwo:Int {
    get {
        return x * 2
    }
    set {
        x = newValue / 2
    }
}

这基本上是Jack Wu的答案,但不同的是,在Jack Wu的答案中,他的x变量是var x: Int,在我的答案中,我的x变量是这样的:var x: Int!,所以我所做的只是使它成为一个可选类型。

Swift中的setter /getter与ObjC有很大不同。属性变成一个计算属性,这意味着它不像在ObjC中那样有一个支持变量,比如_x。

在下面的解决方案代码中,您可以看到xTimesTwo不存储任何东西,而是简单地计算x的结果。

请参阅有关计算属性的官方文档。

您想要的功能也可能是Property observer。

你需要的是:

var x: Int

var xTimesTwo: Int {
    set {
       x = newValue / 2
    }
    get {
        return x * 2
    }
}

您可以在setter/getter中修改其他属性,这就是它们的意义所在。

Swift 5.1更新

在Swift 5.1中,你现在可以不使用get关键字来获取你的变量。例如:

var helloWorld: String {
"Hello World"
}