Возникли проблемы с согласованием модели с перечислениями в Codable

Запись - это сохранение / запись модели и перечисления EmployeeType в файл, но при чтении получает ноль. Я думаю, что ошибка в расширении декодирования перечисления EmployeeType? Я добавил URL-адреса печати, чтобы убедиться, что каталог и файлы для записи и чтения совпадают. Третий вывод должен показать, что возвращенные «decodedEmployees», поступающие из метода чтения / декодирования readEmployeesFromFile (), возвращают nil, что означает, что ошибка находится в «let decodedEmployees = try? ...»

import UIKit
struct Employee: Codable {

    var name: String
    var dateOfBirth: Date
    var employeeType: EmployeeType
}

enum EmployeeType: Codable {
    case exempt
    case nonExempt
    case partTime

    static let all: [EmployeeType] = [.exempt, .nonExempt, .partTime]

    func description() -> String {
        switch self {
        case .exempt:
            return "Exempt Full Time"
        case .nonExempt:
            return "Non-exempt Full Time"
        case .partTime:
            return "Part Time"
        }
    }
}

extension EmployeeType {
    init(from decoder: Decoder) throws {
        var container = try decoder.unkeyedContainer()
        let employeeType = try container.decode(EmployeeType.self)
        self = employeeType

        return
        }
     func encode(to encoder: Encoder) throws {
        var container = encoder.unkeyedContainer()
        try container.encode(description())
        return
    }
}

var count = 0
 func readEmployeeFromFile()->[Employee] {
       let documentsDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).first!
       let archiveURL = documentsDirectory.appendingPathComponent("Employees").appendingPathExtension("plist")
       let employeeDecoder = PropertyListDecoder()
            print("Read: \(archiveURL)")
    if let employeesDataFromFile = try? Data(contentsOf: archiveURL) {
        let decodedEmployees = try? employeeDecoder.decode(Array<Employee>.self, from: employeesDataFromFile)
        print("failed read if nil: \(decodedEmployees)")
    }
    return[Employee(name: "Fail", dateOfBirth: Date(), employeeType: .partTime)]
}

    func writeEmployeeToFile(newEmployeesToSave: [Employee]){
        let documentsDirectory =
            FileManager.default.urls(for: .documentDirectory,
                in: .userDomainMask).first!
            let archiveURL =
            documentsDirectory.appendingPathComponent("Employees")
               .appendingPathExtension("plist")
            print("Write: \(archiveURL)")
        let propertyListEncoder = PropertyListEncoder()
        let savedEmployees = try? propertyListEncoder.encode(newEmployeesToSave)
        try? savedEmployees!.write(to: archiveURL, options: .noFileProtection)
    }

let emps = [Employee(name: "A", dateOfBirth: Date(), employeeType: .exempt)]
writeEmployeeToFile(newEmployeesToSave: emps)
let readEmps = readEmployeeFromFile()
print("read: \(readEmps)")


person DIV    schedule 03.02.2020    source источник


Ответы (1)


Вы слишком усложняете задачу:

enum EmployeeType: String, Codable, CaseIterable {
  case exempt = "Exempt Full Time"
  case nonExempt = "Non-exempt Full Time"
  case partTime = "Part Time"
}

Все остальное будет работать. allCases свойство создается автоматически.

Если вы действительно хотите выполнить кодирование самостоятельно, вам следует изучить singleValueContainer вместо unkeyedContainer и использовать init(rawValue:) для инициализации. Контейнер без ключа - это, по сути, массив.

person Sulthan    schedule 03.02.2020
comment
Расширяя этот ответ, я бы вложил перечисление в структуру Employee и переименовал бы его в Kind. Что-то вроде struct Employee: Codable { let name: String let dateOfBirth: Date let kind: Kind enum Kind: String, Codable, CaseIterable { case exempt = "Exempt Full Time", nonExempt = "Non-exempt Full Time", partTime = "Part Time" } } и при объявлении let employeeKind: Employee.Kind = .exempt - person Leo Dabus; 03.02.2020
comment
спасибо - работает! Не знал о протоколе, допускающем регистрацию. а понятнее добрый enum. - person DIV; 03.02.2020