• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    迪恩网络公众号

ios - 修改对象的内部属性时,仅获取协议(protocol)中定义的属性会导致编译错误

[复制链接]
菜鸟教程小白 发表于 2022-12-11 18:33:12 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题

考虑这样的代码:

protocol SomeProtocol {
    var something: Bool { get set }
}

class SomeProtocolImplementation: SomeProtocol {
    var something: Bool = false {
        didSet {
            print("something changed!")
        }
    }
}

protocol MyProtocol {
    var myProperty: SomeProtocol { get }
}

class MyClass: MyProtocol {
    var myProperty: SomeProtocol = SomeProtocolImplementation() {
        didSet {
            print("myProperty has changed")
        }
    }
}


var o: MyProtocol = MyClass()
o.myProperty.something = true

此代码编译时出错:

error: cannot assign to property: 'myProperty' is a get-only property
o.myProperty.something = true
~~~~~~~~~~~~           ^

为什么?我的属性是 SomeProtocolImplementation 类型,它是类类型,因此应该可以使用对 myProperty 的引用来修改它的内部属性。

更进一步,修改 myProperty 定义使其看起来像这样:

var myProperty: SomeProtocol { get set }

发生了一些奇怪的事情。现在代码编译(并不奇怪),但输出是:

something changed!
myProperty has changed

所以此时 SomeProtocolImplementation 开始表现得像一个值类型 - 修改它的内部状态会导致触发 myProperty 的“didSet”回调。就像 SomeProtocolImplementation 是 struct ...

我实际上找到了解决方案,但我也想了解发生了什么。解决方法是将 SomeProtocol 定义修改为:

protocol SomeProtocol: class {
    var something: Bool { get set }
}

它工作正常,但我试图理解它为什么会这样。谁能解释一下?



Best Answer-推荐答案


首先阅读 Class Only Protocol是。专注于注释部分:

Use a class-only protocol when the behavior defined by that protocol’s requirements assumes or requires that a conforming type has reference semantics rather than value semantics.

上面的引用应该会让你明白。

您正试图为您的 SomeProtocol 的符合类(即 SomeProtocolImplementation)获取 reference type 的行为。您希望将来能够更改 something 的值。所以基本上你是指向上面引用的句子。

如果您需要更多说明,请考虑以下更有意义的设计,为方便起见,我更改了命名:

protocol Base: class {
    var referenceTypeProperty: Bool { get set }
    // By now you are assuming: this property should be modifiable from any reference.
    // So, instantly make the protocol `Class-only`
}

class BaseImplementation: Base {
    var referenceTypeProperty: Bool = false {
        didSet {
            print("referenceTypeProperty did set")
        }
    }
}

protocol Child {
    var valueTypeProperty: Base { get }
    // This property shouldn't be modifiable from anywhere.
    // So, you don't need to declare the protocol as Class-only
}

class ChildImplementation: Child {
    var valueTypeProperty: Base = BaseImplementation() {
        didSet {
            print("valueTypeProperty did set")
        }
    }
}

let object: Child = ChildImplementation()
object.valueTypeProperty.referenceTypeProperty = true

关于ios - 修改对象的内部属性时,仅获取协议(protocol)中定义的属性会导致编译错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45836642/

回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关注0

粉丝2

帖子830918

发布主题
阅读排行 更多
广告位

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap