Swift 3에서 사용자 지정 개체를 저장할 때 비속성 목록 개체 삽입 시도
나는 다음과 일치하는 간단한 물체를 가지고 있습니다.NSCoding
의전
import Foundation
class JobCategory: NSObject, NSCoding {
var id: Int
var name: String
var URLString: String
init(id: Int, name: String, URLString: String) {
self.id = id
self.name = name
self.URLString = URLString
}
// MARK: - NSCoding
required init(coder aDecoder: NSCoder) {
id = aDecoder.decodeObject(forKey: "id") as? Int ?? aDecoder.decodeInteger(forKey: "id")
name = aDecoder.decodeObject(forKey: "name") as! String
URLString = aDecoder.decodeObject(forKey: "URLString") as! String
}
func encode(with aCoder: NSCoder) {
aCoder.encode(id, forKey: "id")
aCoder.encode(name, forKey: "name")
aCoder.encode(URLString, forKey: "URLString")
}
}
나는 그것의 예를 저장하려고 노력하고 있습니다.UserDefaults
하지만 계속 실패하고 다음 오류가 발생합니다.
감지되지 않은 예외 'NSInvalidArgument'로 인해 앱을 종료합니다.예외', 이유: '키 작업 범주에 대한 비속성 목록 개체를 삽입하려고 합니다.
저장할 코드입니다.UserDefaults
.
enum UserDefaultsKeys: String {
case jobCategory
}
class ViewController: UIViewController {
@IBAction func didTapSaveButton(_ sender: UIButton) {
let category = JobCategory(id: 1, name: "Test Category", URLString: "http://www.example-job.com")
let userDefaults = UserDefaults.standard
userDefaults.set(category, forKey: UserDefaultsKeys.jobCategory.rawValue)
userDefaults.synchronize()
}
}
enum 값을 key로 바꾸었지만 동일한 오류가 발생합니다.원인이 뭔지 아세요?
다음을 생성해야 합니다.Data
예를 들어JobCategory
사용하여 모델링 및 저장Data
의 예.UserDefaults
를 사용하여 나중에 디코딩합니다.
struct JobCategory: Codable {
let id: Int
let name: String
}
// To store in UserDefaults
if let encoded = try? JSONEncoder().encode(category) {
UserDefaults.standard.set(encoded, forKey: UserDefaultsKeys.jobCategory.rawValue)
}
// Retrieve from UserDefaults
if let data = UserDefaults.standard.object(forKey: UserDefaultsKeys.jobCategory.rawValue) as? Data,
let category = try? JSONDecoder().decode(JobCategory.self, from: data) {
print(category.name)
}
오래된 대답
다음을 생성해야 합니다.Data
예를 들어JobCategory
인스턴스 사용 및 저장Data
의 예.UserDefaults
나중에 를 사용하여 보관을 해제합니다. 이렇게 하십시오.
데이터 저장 위치UserDefaults
let category = JobCategory(id: 1, name: "Test Category", URLString: "http://www.example-job.com")
let encodedData = NSKeyedArchiver.archivedData(withRootObject: category, requiringSecureCoding: false)
let userDefaults = UserDefaults.standard
userDefaults.set(encodedData, forKey: UserDefaultsKeys.jobCategory.rawValue)
다음에서 데이터 검색용UserDefaults
let decoded = UserDefaults.standard.object(forKey: UserDefaultsKeys.jobCategory.rawValue) as! Data
let decodedTeams = NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(decoded) as! JobCategory
print(decodedTeams.name)
Swift 4, Xcode 10 업데이트
쉽게 접근할 수 있도록 주변에 구조물을 작성했습니다.
//set, get & remove User own profile in cache
struct UserProfileCache {
static let key = "userProfileCache"
static func save(_ value: Profile!) {
UserDefaults.standard.set(try? PropertyListEncoder().encode(value), forKey: key)
}
static func get() -> Profile! {
var userData: Profile!
if let data = UserDefaults.standard.value(forKey: key) as? Data {
userData = try? PropertyListDecoder().decode(Profile.self, from: data)
return userData!
} else {
return userData
}
}
static func remove() {
UserDefaults.standard.removeObject(forKey: key)
}
}
프로필이 Json 인코딩 개체입니다.
struct Profile: Codable {
let id: Int!
let firstName: String
let dob: String!
}
용도:
//save details in user defaults...
UserProfileCache.save(profileDetails)
그게 도움이 되길 바랍니다!!!
감사해요.
빠른 세이브Codable
에 반대하는.UserDefault
와 함께@propertyWrapper
@propertyWrapper
struct UserDefault<T: Codable> {
let key: String
let defaultValue: T
init(_ key: String, defaultValue: T) {
self.key = key
self.defaultValue = defaultValue
}
var wrappedValue: T {
get {
if let data = UserDefaults.standard.object(forKey: key) as? Data,
let user = try? JSONDecoder().decode(T.self, from: data) {
return user
}
return defaultValue
}
set {
if let encoded = try? JSONEncoder().encode(newValue) {
UserDefaults.standard.set(encoded, forKey: key)
}
}
}
}
enum GlobalSettings {
@UserDefault("user", defaultValue: User(name:"",pass:"")) static var user: User
}
예제 사용자 모델 코드화 가능 확인
struct User:Codable {
let name:String
let pass:String
}
사용방법
//Set value
GlobalSettings.user = User(name: "Ahmed", pass: "Ahmed")
//GetValue
print(GlobalSettings.user)
사전을 사용자 기본값으로 저장
let data = NSKeyedArchiver.archivedData(withRootObject: DictionaryData)
UserDefaults.standard.set(data, forKey: kUserData)
사전 검색
let outData = UserDefaults.standard.data(forKey: kUserData)
let dict = NSKeyedUnarchiver.unarchiveObject(with: outData!) as! NSDictionary
하르조트 싱의 답변을 기반으로 합니다.나는 이렇게 사용했습니다.
struct AppData {
static var myObject: MyObject? {
get {
if UserDefaults.standard.object(forKey: "UserLocationKey") != nil {
if let data = UserDefaults.standard.value(forKey: "UserLocationKey") as? Data {
let myObject = try? PropertyListDecoder().decode(MyObject.self, from: data)
return myObject!
}
}
return nil
}
set {
UserDefaults.standard.set(try? PropertyListEncoder().encode(newValue), forKey: "UserLocationKey")
}
}
}
여기에UserDefaults
설정 및 가져오기 확장Codable
개체를 일반 텍스트 파일로 여는 경우 이 개체를 플리스트(사용자 기본값)에 사용자가 읽을 수 있도록 유지합니다.
extension Encodable {
var asDictionary: [String: Any]? {
guard let data = try? JSONEncoder().encode(self) else { return nil }
return try? JSONSerialization.jsonObject(with: data) as? [String : Any]
}
}
extension Decodable {
init?(dictionary: [String: Any]) {
guard let data = try? JSONSerialization.data(withJSONObject: dictionary) else { return nil }
guard let object = try? JSONDecoder().decode(Self.self, from: data) else { return nil }
self = object
}
}
extension UserDefaults {
func setEncodableAsDictionary<T: Encodable>(_ encodable: T, for key: String) {
self.set(encodable.asDictionary, forKey: key)
}
func getDecodableFromDictionary<T: Decodable>(for key: String) -> T? {
guard let dictionary = self.dictionary(forKey: key) else {
return nil
}
return T(dictionary: dictionary)
}
}
plist 배열 간의 배열(코드 가능한 배열)도 지원하려면 확장에 다음을 추가합니다.
extension UserDefaults {
func setEncodablesAsArrayOfDictionaries<T: Encodable>(_ encodables: Array<T>, for key: String) {
let arrayOfDictionaries = encodables.map({ $0.asDictionary })
self.set(arrayOfDictionaries, forKey: key)
}
func getDecodablesFromArrayOfDictionaries<T: Decodable>(for key: String) -> [T]? {
guard let arrayOfDictionaries = self.array(forKey: key) as? [[String: Any]] else {
return nil
}
return arrayOfDictionaries.compactMap({ T(dictionary: $0) })
}
}
목록이 사람이 읽을 수 있는 것에 대해 신경 쓰지 않는다면 간단히 저장할 수 있습니다.Data
(일반 텍스트로 열 경우 임의 문자열처럼 표시됨):
extension UserDefaults {
func setEncodable<T: Encodable>(_ encodable: T, for key: String) throws {
let data = try PropertyListEncoder().encode(encodable)
self.set(data, forKey: key)
}
func getDecodable<T: Decodable>(for key: String) -> T? {
guard
self.object(forKey: key) != nil,
let data = self.value(forKey: key) as? Data
else {
return nil
}
let obj = try? PropertyListDecoder().decode(T.self, from: data)
return obj
}
}
(이 두 번째 접근 방식을 사용하면 다음과 같은 이점이 필요하지 않습니다.Encodable
그리고.Decodable
맨 위에서 확장)
언급URL : https://stackoverflow.com/questions/41355427/attempt-to-insert-non-property-list-object-when-trying-to-save-a-custom-object-i
'programing' 카테고리의 다른 글
노드JS - setTimeout(fn,0) vs setImediate(fn) (0) | 2023.08.21 |
---|---|
PowerShell로 폴더를 검색하는 방법 (0) | 2023.08.21 |
서비스 계층이 저장소 구현에 어떻게 적합합니까? (0) | 2023.08.21 |
스프링 부트로 여러 서블릿 컨테이너/서블릿 구성 (0) | 2023.08.21 |
16진수 색상 문자열에서 색상을 가져오는 방법 (0) | 2023.08.21 |