微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

swift – 错误:尝试将堆栈放在不可读的内存中:

我正在尝试向UIViewController添加其他属性.

码:

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)}
    }
}

但是我收到以下错误

错误:尝试将堆栈放在不可读的内存中:

我知道我们不能直接将存储的属性添加到扩展,所以我尝试使用objc_setAssociatedobject()添加

解决方法

你正在做的事情有很多问题:

>尝试在自己的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] 举报,一经查实,本站将立刻删除。

相关推荐