“群芳争艳”:CoreData 4 种方法计算最大值的效率比较(上)

在这里插入图片描述

概览

在 CoreData 支持的 App 中,一种常见操作就是计算数据库表中指定字段的最大值(或最小值)。就是这样一种看起来“不足挂齿”的任务,可能稍不留神就会“马失前蹄”。
在这里插入图片描述

在实际的代码中,我们怎样才能既迅速又简洁的找出字段的最大值呢?

在本篇博文中,您将学到如下内容:

  • 概览
  • 0. CoreData 表结构
  • 1. 借助现成的关系属性(Relation Property)
  • 2. 使用“远古” NSArray 的力量
  • 3. 使用 CoreData Fetch 请求
  • 总结

相信学完本课后,大家 CoreData 的算法武器库中又会多几种“削铁如泥”的利刃啦。

本文中所有代码的测试环境为:

  • MBA 2022,M2,内存 16 GB
  • macOS 15.3.2(Sequoia)

那还等什么呢?Let‘s find out!!!😉


0. CoreData 表结构

我们即将展示的所有算法都将在如下 CoreData 表结构中“施展拳脚”:

  • Project 表包含多个 VictoryStage 对象,这通过 Project 的 records 关系体现出来;
  • VictoryStage 表中包含一个类型为 Date? 的 end 字段,它用来表示 VictoryStage 的结束日期;

数据库中被测试的特定 Project 对象里大约包含不到 400 个 VictoryStage 对象,我们要计算的就是:这些对象中 end 字段最大的值

为了方便,我们首先创建一个将集合(Set)类型 records 转换为数组(Array)的辅助方法:

extension Project {var vstages: [VictoryStage] {if let recordAry = records?.allObjects as? [VictoryStage] {return recordAry}return []}
}

接下来,小伙伴们就可以“驰马试剑”,恣意测试我们的算法啦!

1. 借助现成的关系属性(Relation Property)

既然 Project 包含一对多的 vstages 关系,我们最简单的方法是直接从此入手,找到 end 值最大(时间上最新)的那一个 VictoryStage 对象。


因为最新的 Swift Testing 框架还缺少内置的测速机制,我们只能暂时退回到 XCTest 的 measure 方法中去。这里需要提出批评,希望苹果赶快在 Swift Testing 2 中将其补上哦。

关于更多驯服 Swift Testing 测试的小妙招,请小伙伴们移步如下链接观赏精彩的内容:

  • 用接地气的例子趣谈 WWDC 24 全新的 Swift Testing 入门(一)
  • 用接地气的例子趣谈 WWDC 24 全新的 Swift Testing 入门(二)
  • 用接地气的例子趣谈 WWDC 24 全新的 Swift Testing 入门(三)
  • WWDC24(Xcode 16)中全新的 Swift Testing 使用进阶
  • Xcode 16 中 Swift Testing 的参数化(Parameterized)机制趣谈

算法的实现很简单:我们只需将 vstages 数组按照 end 从新到旧排序,结果数组中第一个 VictoryStage 对象就是我们想要的。

func testPerformanceWithRelationProperty() throws {// 利用 Project 关系属性来计算最新的 VictoryStagemeasure {let vstages = project.vstageslet mostRecent = vstages.sorted(using: SortDescriptor(\VictoryStage.end, order: .reverse)).firstprint("最新的 VStage: \(mostRecent?.end ?? Date.distantPast)")}
}

从单元测试的结果来看,10 次运行的平均耗时为:0.00464 秒。

在这里插入图片描述

2. 使用“远古” NSArray 的力量

我们知道,Swift 中的数组(Array)都是由底层的 Objc 数据结构 NSArray 在默默“撑腰”的。而 NSArray 又自带了一个 sortedArray 排序方法:

在这里插入图片描述
So,我们可以很快据此写出 NSArray 的排序算法:

func testPerformanceWithRelationPropertyNSArray() throws {// 利用 Project 关系属性的 NSArray 来计算最新的 VictoryStagemeasure {let vstages = project.vstageslet mostRecent = ((vstages as NSArray).sortedArray(using: [.init(keyPath: \VictoryStage.end, ascending: false)]) as! [VictoryStage]).firstprint("最新的 VStage: \(mostRecent?.end ?? Date.distantPast)")}
}

美中不足的是,这种方法需要做 2 次类型转换。

小伙伴们可能觉得 NSArray 排序算法和前一种速度差不了多少,不过实际结果可能会让你们大吃一惊:

在这里插入图片描述

是的,你们没有看错!NSArray 排序比 Array 排序快了一个数量级,只需 0.000559 秒。

3. 使用 CoreData Fetch 请求

为了能够充分发挥 CoreData 底层里 Sqlite 原生查询的“奥义”,我们可以直接利用 Fetch 请求来排忧解难:

func testPerformanceWithFetchPredicate() throws {// 利用 CoreData Fetch 请求来计算最新的 VictoryStagemeasure {let req = VictoryStage.fetchRequest()req.predicate = .init(format: "project = %@", project)req.sortDescriptors = [.init(keyPath: \VictoryStage.end, ascending: false)]req.fetchLimit = 1let mostRecent = try! context.fetch(req).firstprint("最新的 VStage: \(mostRecent?.end ?? Date.distantPast)")}
}

在上面的代码中,我们做了这样几件事:

  1. 创建一个 Fetch Request;
  2. 稳妥的设置了它的 Predicate 和排序属性;
  3. 将其搜索结果的数量限制为 1 ;

运行看一下效果!恭喜大家,我们又成功的将性能提升了几倍,现在只需 0.000196 秒了。

在这里插入图片描述

在下一篇文章中,我们将再接再厉,使用 NSExpression 表达式方法来为本系列博文画上一个完美的句号。


想要进一步系统地学习 Swift 开发的小伙伴们,可以来我的《Swift 语言开发精讲》专栏逛一逛哦:

在这里插入图片描述

  • 《Swift 语言开发精讲》

总结

在本篇博文中,我们讨论了 CoreData 中计算字段最大值的三种方法,任君选用。

感谢观赏,我们下一篇再见!😎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/75864.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

skynet网络包库(lua-netpack.c)的作用解析

目录 网络包库(lua-netpack.c)的作用解析1. 数据包的分片与重组2. 网络事件处理3. 内存管理4. 数据打包与解包 动态库(.so)在 Lua 中的使用1. 编译为动态库2. Lua 中加载与调用(1) 加载模块(2) 核心方法(3) 使用示例 3. 注意事项 …

计科数据库第二次上机操作--实验二 表的简单查询

一、建数据库和表 1.启动数据库服务软件 Navicat 2.在 Navicat 中建立数据库 test 3. 在test数据库上建立teacher表: 二、基本查询 2.1 从teacher表中分别检索出教师的所有信息 SELECT * FROM teacher WHERE 教工号2000; SELECT * FROM t…

WPF依赖注入

一、IOC 在 WPF 中的原理 控制反转(IOC)是一种设计原则,它将对象的创建和依赖关系的管理从对象本身转移到外部容器(IOC 容器)。在传统的编程方式中,一个对象如果需要使用另一个对象(即存在依赖…

【大模型深度学习】如何估算大模型需要的显存

一、模型参数量 参数量的单位 参数量指的是模型中所有权重和偏置的数量总和。在大模型中,参数量的单位通常以“百万”(M)或“亿”(B,也常说十亿)来表示。 百万(M):表示…

BUUCTF流量分析题

文章目录 前言wireshark被嗅探的流量被偷走的文件easycap数据包中的线索秘密文件[安洵杯 2019]Attack (难,没写)被劫持的神秘礼物大流量分析(一)大流量分析(二)大流量分析(三)模板模板 前言 CT…

adb检测不到原来的设备List of devices attached解决办法

进设备管理器-通用串行总线设备 卸载无法检测到的设备驱动 重新拔插数据线

mapbox基础,加载栅格图片到地图

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性1.3 ☘️raster 栅格图层 api二、🍀使用本地载…

复活之我会二分

文章目录 整数二分模板模板1:满足条件的第一个数模板2:满足条件的最后一个数 浮点数二分模板一、Building an Aquarium思路分析具体代码 二、Tracking Segments思路分析具体代码 三、Wooden Toy Festival思路分析具体代码 四、路标设置思路分析具体代码 …

每日c/c++题 备战蓝桥杯(握手问题)

试题 A: 握手问题 题解 题目描述 小蓝组织了一场算法交流会议,共有50人参加。按照惯例,每个人都要与除自己外的其他所有人握手一次。但有7个人彼此之间没有握手(这7人与其他43人正常握手)。求实际发生的握手总次数。 解题思路 …

mysql8.0.29 win64下载

mysql win64安装包 mysql win64安装包下载 mysql win64安装包下载 通过网盘分享的文件:mysql 链接: https://pan.baidu.com/s/1sEOl-wSVtOG5gfIRdt5MXw?pwdgi7i 提取码: gi7i

browser-use开源程序使 AI 代理可以访问网站,自动完成特定的指定任务,告诉您的计算机该做什么,它就会完成它。

一、软件介绍 文末提供程序和源码下载 browser-use开源程序使 AI 代理可以访问网站,自动完成特定的指定任务,浏览器使用是将AI代理与浏览器连接的最简单方法。告诉您的计算机该做什么,它就会完成它。 二、快速开始 使用 pip (Py…

CAD格式转换器:Acme CAD Converter

Acme CAD Converter 是一款专业的多功能 CAD 文件管理工具,支持 ‌DWG/DXF/DWF 文件查看、批量格式转换及版本降级‌,适用于工程设计、图纸归档等场景‌。软件兼容 AutoCAD R2.5 至 2023 版本文件‌,可输出为 PDF、JPEG、TIFF、SVG 等 20 格式…

vmware虚拟机上Ubuntu或者其他系统无法联网的解决方法

一、检查虚拟机是否开启了网络服务 打开方式:控制面板->-管理工具--->服务 查找 VMware DHCP Service 和VMware NAT Service ,确保这两个服务已经启动。如下图,没有启动就点击启动。 二、设置网络类型 我们一般使用前两种多一些&…

数据结构与算法:基础与进阶

🌟 各位看官好,我是maomi_9526! 🌍 种一棵树最好是十年前,其次是现在! 🚀 今天来学习C语言的相关知识。 👍 如果觉得这篇文章有帮助,欢迎您一键三连,分享给更…

使用分布式锁和乐观锁解决超卖问题

在电商、秒杀等高并发场景中,“超卖”问题指库存被过量扣减,导致实际库存不足。以下是使用 分布式锁 和 乐观锁 解决超卖问题的原理与实现方案: 一、超卖问题的核心原因 多个并发请求同时读取库存余量,并在本地计算后发起写操作&…

盛水最多的容器

本题有两种解法,一种是暴力解法,直接暴力枚举出所有的体积比较出最大的即可,但是时间复杂度达到n方。超出了限制,另一种解法就是利用单调性解法,我们着重介绍一下单调性解法。 单调性解法: 体积vh*w&…

操作系统概述(3)

批处理系统 1.单道批处理系统 单道批处理系统是成批地处理作用,并且始终只有一道作业在内存中的系统。优点:提高系统资源的利用率和系统吞吐量。缺点:系统中的资源得不到充分利用。 2.多道批处理系统 引入多道程序设计技术,是…

数字身份DID协议:如何用Solidity编写去中心化身份合约

本文提出基于以太坊的自主主权身份(SSI)实现方案,通过扩展ERC-734/ERC-735标准构建链上身份核心合约,支持可验证声明、多密钥轮换、属性隐私保护等特性。设计的三层架构体系将身份控制逻辑与数据存储分离,在测试网环境…

【目标检测】【深度学习】【Pytorch版本】YOLOV2模型算法详解

【目标检测】【深度学习】【Pytorch版本】YOLOV2模型算法详解 文章目录 【目标检测】【深度学习】【Pytorch版本】YOLOV2模型算法详解前言YOLOV2的模型结构YOLOV2模型的基本执行流程YOLOV2模型的网络参数YOLOV2模型的训练方式 YOLOV2的核心思想前向传播阶段反向传播阶段 总结 前…

第421场周赛:数组的最大因子得分、

Q1、数组的最大因子得分 1、题目描述 给你一个整数数组 nums。 因子得分 定义为数组所有元素的最小公倍数(LCM)与最大公约数(GCD)的 乘积。 在 最多 移除一个元素的情况下,返回 nums 的 最大因子得分。 注意&…