Возникли проблемы с соответствием модели с перечислениями Codable - PullRequest
0 голосов
/ 03 февраля 2020

Запись - это сохранение / запись модели и перечисления 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)")

1 Ответ

1 голос
/ 03 февраля 2020

Вы делаете это слишком сложно:

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:) для инициализации. Контейнер без ключа представляет собой массив.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...