- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
在 Swift 4 中以自定义方式将自定义类型编码为 JSON 很容易,并且有很好的文档可用。但是我在以自定义方式编码 Foundation
类型时遇到了问题。我的具有类型 TimeInterval
属性的自定义类型需要是 Codable
并使用 JSON 格式显示分钟和秒,如下所示:
// my type:
struct MyType: Codable {
let time: TimeInterval = 65.12
}
// required JSON representation:
"myType": {
"time":
{
"minutes": 1,
"seconds": 5.12
}
}
如果我为我的每一个类型提供 encode(to:)
和 init(decoder:)
的自定义实现,这很好用。但我不想重复我的话,只是为 TimeInterval
提供这些方法的自定义版本。所以我尝试了这样的扩展:
extension TimeInterval {
enum CodingKeys: String, CodingKey {
case minutes
case seconds
}
func encode(to encoder: Encoder) throws {
let minutes = (NSInteger(self) / 60) % 60
let seconds = self.truncatingRemainder(dividingBy: 60)
var container = encoder.container(keyedBy: CodingKeys.self)
try container.encode(minutes, forKey: .minutes)
try container.encode(seconds, forKey: .seconds)
}
init(from decoder: Decoder) throws {
let values = try decoder.container(keyedBy: CodingKeys.self)
guard let minutes = try? values.decode(Int.self, forKey: .minutes), let seconds = try? values.decode(Double.self, forKey: .seconds) else {
throw NSError()
}
self = TimeInterval(Double(minutes) * 60.0 + seconds)
}
}
不幸的是,这段代码永远不会被调用:编码一个 TimeInterval
值似乎总是使用编码一个 Double
的默认实现,它符合 Codable
开箱即用。
let myType = MyType(time: 65.12)
let encoded = try! JSONEncoder().encode(myType)
let json = String(data: encoded, encoding: .utf8)!
print(json)
// prints:
// "myType":
// {
// "time": 65.12
// }
我该如何解决这个问题?
编辑:就像我在第一句话中写的那样,我不是在寻找解决方案,如何编码我自己的自定义类型。这在我的代码中已经运行良好。我正在尝试更改基础类型 TimeInterval
的编码行为。
最佳答案
您可以将您的时间间隔设置为只读计算属性,并添加一个采用 TimeInterval 的自定义初始化程序并初始化您的时间子结构属性:
struct Root: Codable {
let myType: MyType
}
struct MyType: Codable {
let time: Time
struct Time: Codable {
let minutes: Int
let seconds: Double
}
init(time: TimeInterval) {
self.time = Time(minutes: (Int(time) / 60) % 60, seconds: time.truncatingRemainder(dividingBy: 60))
}
var timeInterval: TimeInterval {
return TimeInterval(time.minutes * 60) + time.seconds
}
}
let root = Root(myType: MyType(time: 65.12))
do {
let encoded = try JSONEncoder().encode(root)
print(String(data: encoded, encoding: .utf8)!) // "{"myType":{"time":{"minutes":1,"seconds":5.1200000000000045}}}\n
} catch {
print(error)
}
编辑/更新:
如果您需要将时间属性添加到多个结构中,您可以将结构时间从自定义类型中移出并创建一个时间协议(protocol):
struct Time: Codable {
let minutes: Int
let seconds: Double
}
protocol Timed {
var time: Time { get }
}
扩展 Timed 协议(protocol),添加 timeInterval 只读计算属性
extension Timed {
var timeInterval: TimeInterval {
return TimeInterval(time.minutes * 60) + time.seconds
}
}
添加一个自定义的初始化器来延长时间:
extension Time {
init(time: TimeInterval) {
self.minutes = (Int(time) / 60) % 60
self.seconds = time.truncatingRemainder(dividingBy: 60)
}
}
然后您只需要将 Timed 协议(protocol)添加到具有时间属性的结构中:
struct Root: Codable {
let myType: MyType
}
struct MyType: Codable, Timed {
let time: Time
}
测试:
let root = Root(myType: MyType(time: Time(time: 65.12)))
root.myType.timeInterval
do {
let encoded = try JSONEncoder().encode(root)
print(String(data: encoded, encoding: .utf8)!) // "{"myType":{"time":{"minutes":1,"seconds":5.1200000000000045}}}\n
} catch {
print(error)
}
关于json - 如何以自定义方式对已经符合 Codable 的基础类型(例如 TimeInterval)进行编码?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47760983/
vue3 快速入门系列 - 基础 前面我们已经用 vue2 和 react 做过开发了。 从 vue2 升级到 vue3 成本较大,特别是较大的项目。所以许多公司对旧项目继续使用vue2,新项目则
C# 基础 C#项目创建 这里注意win10虚拟机需要更新下补丁,不然直接下载visual studio 2022会显示版本不支持 HelloWorld C#的类文件都是以.cs结尾,入口方法为sta
关于 iPhone 内存管理的非常基本的问题: 假设我有一个 viewController,其中有几个 subview 也由 viewController 控制。当我删除顶部 viewControll
我仍在努力适应指针。不是概念——我理解内存位置、匹配可变长度的指针增量等——这是语法。这是一个我认为是我感到困惑/无法直观把握的原因之一: int a = 42; 在一个int大小的内存空间中分配并放
1. 简介 Kafka(Apache Kafka) 是一种分布式流数据平台,最初由LinkedIn开发,并于后来捐赠给Apache软件基金会,成为了一个Apache顶级项目。它被设计用于处理大规
1.想要在命令提示符下操作mysql服务器,添加系统变量。(计算机-系统属性——环境变量——path) 2.查询数据表中的数据; select selection_lis
MySQL表的增删改查(基础) 1. CRUD 注释:在SQL中可以使用“–空格+描述”来表示注释说明 CRUD 即增加(Create)、查询(Retrieve)、更新(Update)、删除(Dele
我有一个网页,可以在加载时打开显示模式,在这个模式中,我有一个可以打开第二个模式的链接。当第二个模式关闭时(通过单击关闭按钮或单击模式外部),我想重新打开第一个模式。 对于关闭按钮,我可以通过向具有
使用 Core Data Fetched Properties,我如何执行这个简单的请求: 我希望获取的属性 ( myFetchProp ) 存储 StoreA ,它应该这样做: [myFetchPr
关闭。这个问题是opinion-based .它目前不接受答案。 想改进这个问题?更新问题,以便 editing this post 可以用事实和引用来回答它. 8年前关闭。 Improve this
最近,我得到了一个现有的Drupal项目,并被要求改进前端(HTML,JavaScript,CSS)。我在Django,PHP,Ruby等方面具有大量的前端和后端开发经验,但是我没有任何Drupal经
我试图让我的用户通过使用扫描仪类来决定要做什么,但我有一个问题,代码一旦运行就不会激活,并且它不会让我跳过任何行。我的代码如下所示: Scanner input = new Scanner(S
对模糊的标题表示歉意,因为我想不出这个名字是什么。 基本上创建一个计算学生财务付款的小程序。当我运行它时,它计算对象限额没有问题。然而,无论我尝试什么,对象“助学金”似乎除了 0 之外什么也没有提出。
这是我的代码 - main() { double x; double y = pow(((1/3 + sin(x/2))(pow(x, 3) + 3)), 1/3); prin
如果我的术语在这个问题上有误,我们深表歉意。 采取以下功能: i = 1; v = i * 2; for (j = 0; j < 4; j++ ) { console.log(v);
我的应用程序中有不同的类文件。我有 5 个类,其中 2 个是 Activity ,1 个是运行的服务。其他 2 个只是类。这两个类中变量的生命周期是多少。我知道一个 Activity 可以被操作系统杀
例如,一个方法返回一个 List 类型的对象。 public List bojangles () ... 一些代码调用方法FooBar.bojangles.iterator(); 我是 Java 的新
我遇到了一个奇怪的问题,网格的大小不适合我的屏幕。当我使用 12 列大时,它只占据屏幕的 1/3 的中间,请参见图像。我不确定是什么导致了这个问题。我没有任何会导致这种情况发生的奇怪 CSS。我不会在
我尝试使用头文件和源文件,但遇到了问题。因此,我对我正在尝试做的事情做了一个简化版本,我在 CodeBlocks 中遇到了同样的错误(undefined reference to add(double
我正在为我的网格系统使用基础,但这在任何网格系统中都可能是一个问题。我基本上用一个容器包裹了 3 个单元格,但其中一个单元格应该长到页面边框(留在我的 Sampe-Image 中)但这也可能在右侧)。
我是一名优秀的程序员,十分优秀!