我有以下结构:
struct Identity {
var id: Int
var createdAt: NSDate
var string: String
var apnsToken: String
}
在我的应用程序执行过程中,实例(?)将此结构转换为NSData
(使用以下代码)并存储在NSUserDefaults
中
var id = Identity(id: 0, createdAt: NSDate(), string: "string", apnsToken: "<apns-token-here>")
var data = NSData(bytesNoCopy: &id, length: sizeof(Identity),freeWhenDone:false)
当我试图从NSData
实例获取一个结构时,它会使用一个EXC_BAD_ACCESS (它是一个代码1)崩溃:
var id = UnsafePointer<Identity>(userDefaultsData.bytes).memory
但是,只有当我从NSData
中获得NSUserDefaults实例时,才会发生这种情况。如果我做了下面这样的事情,它在没有崩溃的情况下工作。
var id = Identity(id: 0, createdAt: NSDate(), string: "string", apnsToken: "<apns-token-here>")
var data = NSData(bytesNoCopy: &id, length: sizeof(Identity),freeWhenDone:false)
var idPrime = UnsafePointer<Identity>(data.bytes).memory
EXC_BAD_ACCESS转储的程序集代码在objc_retain的一半处,在and
指令之后。
更新:
我不是完全诚实的。数据从ObjC中的密钥链中检索,bridge_transfer从CF数据对象转换为NSData
。CF对象作为out param来自SecItemCopy()
。我以为NSUserDefaults
会更有关系。
发布于 2015-04-28 04:04:27
这是因为这一行:
var data = NSData(bytesNoCopy: &id, length: sizeof(Identity),freeWhenDone:false)
不会将String
(或任何分配自己内存(如数组)的其他类型)呈现为字节形式。相反,它所要做的就是将指向字符串的内存的指针序列化为NSData
字节。
如果记忆不再存在,这显然会导致爆炸。这就是为什么当你一次完成所有的工作时,它似乎是有效的,而不是当你存储到用户的缺省值,然后,然后,甚至在一个不同的过程中,把它拿出来。
相反,您需要做一些事情,比如将字符串存储到自己的NSData
对象中(比如使用NSData(base64EncodedString:options:)
),然后也存储它。
https://stackoverflow.com/questions/29908816
复制相似问题