码:
protocol AdditionalStoredProperties { associatedtype Title func getAssociatedobject<Title>(key: UnsafePointer<Title>,defValue : Title)->Title } extension AdditionalStoredProperties { func getAssociatedobject<Title>( key: UnsafePointer<Title>,defValue : Title)->Title { guard let actual_value = objc_getAssociatedobject(self as! AnyObject,key) as? Title else { return defValue } return actual_value } } extension UIViewController:AdditionalStoredProperties { typealias Title = String var prev@R_502_6257@sPage : String { get { return getAssociatedobject(&self.prev@R_502_6257@sPage,defValue: self.prev@R_502_6257@sPage) } set { objc_setAssociatedobject(self,&self.prev@R_502_6257@sPage,newValue,.OBJC_ASSOCIATION_RETAIN)} } }
但是我收到以下错误:
错误:尝试将堆栈放在不可读的内存中:
解决方法
>尝试在自己的getter中访问self.prev@R_502_6257@sPage将递归调用自身.
>您不能将& self.prev@R_502_6257@sPage用作稳定或唯一的指针值,因为它将是指向临时变量的指针(因为您正在处理计算属性).因此,您不能将其用作关联对象的键. Swift仅为静态和全局存储变量保证稳定且唯一的指针值(有关详细信息,请参阅this Q&A).
>您应该使AdditionalStoredProperties成为一个类绑定协议(带有:class),因为您只能将关联对象添加到Objective-C类(在Apple平台上,Swift类是基于它构建的).虽然你可以将一个结构桥接到AnyObject(它将被封装在一个不透明的Obj-C兼容包装器中),但它只是那个;一座桥.无法保证您将获得相同的实例,因此无法保证关联的对象将保持不变.
>您可能并不认为Title是您协议的关联类型;你没有用它做任何事情(getAssociatedobject(key:defValue :)定义的通用占位符Title完全不相关).
记住这些要点,这是您的代码的固定版本:
protocol AdditionalStoredProperties : class { func getAssociatedobject<T>(ofType: T.Type,key: UnsafeRawPointer,defaultValue: @autoclosure () -> T) -> T } extension AdditionalStoredProperties { func getAssociatedobject<T>(ofType: T.Type,defaultValue: @autoclosure () -> T) -> T { // or: return objc_getAssociatedobject(self,key) as? T ?? defaultValue() guard let actualValue = objc_getAssociatedobject(self,key) as? T else { return defaultValue() } return actualValue } } extension UIViewController : AdditionalStoredProperties { private enum AssociatedobjectKeys { static var prev@R_502_6257@sPage: Never? } var prev@R_502_6257@sPage: String { get { // return the associated object with a default of "" (feel free to change) return getAssociatedobject(ofType: String.self,key: &AssociatedobjectKeys.prev@R_502_6257@sPage,defaultValue: "") } set { objc_setAssociatedobject(self,&AssociatedobjectKeys.prev@R_502_6257@sPage,.OBJC_ASSOCIATION_RETAIN) } } }
请注意,我们是:
>使用静态存储属性以获取指针值以用作关联对象的键.再次,这是因为Swift guarantees stable and unique pointer values for static and global stored variables.>使用@autoclosure作为defaultValue:参数,因为如果已存在关联对象,则可能不需要对其进行求值.>使用key:参数获取UnsafeRawPointer,因为指针对象的类型无关紧要;它只是内存中用作密钥的位置.>使用ofType:参数明确地满足通用占位符.这主要是一个偏好问题,但我更喜欢明确地拼写这些东西,而不是依赖于类型推断.>使用camelCase而不是snake_case,就像Swift惯例一样.
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。