Swift 的 KeyPath 是什么?

Swift 的 KeyPath 是什么?

      • 一、语法解析
      • 二、KeyPath 的核心作用
        • 1. 类型安全的属性引用
        • 2. 动态访问属性
        • 3. 函数式编程与数据驱动
      • 三、SwiftUI 中的典型应用场景
        • 1. 动态 UI 组件配置
        • 2. 与 `@dynamicMemberLookup` 结合
        • 3. 动画与状态管理
      • 四、对比其他技术
      • 五、进阶技巧
        • 1. 类型擦除的 KeyPath
        • 2. KeyPath 组合
      • 总结

在 Swift 中,KeyPath<Hike.Observation, Range<Double>> 是一种类型安全的属性路径声明,用于以类型安全的方式引用某个类型的属性。这种写法结合了 Swift 的泛型和 KeyPath 特性,具有明确的编译时类型检查能力。


一、语法解析

var path: KeyPath<Hike.Observation, Range<Double>> 的含义是:

  • 根类型Hike.Observation
  • 目标类型Range<Double>
  • 作用:声明一个 KeyPath,表示从 Hike.Observation 类型中某个返回 Range<Double> 类型的属性路径。

例如,如果 Hike.Observation 有一个 elevationRange 属性:

extension Hike.Observation {var elevationRange: Range<Double> { ... }
}

则对应的 KeyPath 可以写作 \Hike.Observation.elevationRange


二、KeyPath 的核心作用

1. 类型安全的属性引用

KeyPath 允许在编译时验证属性是否存在且类型匹配。例如:

// 正确:属性存在且类型匹配
let path: KeyPath<Hike.Observation, Range<Double>> = \.elevationRange// 编译错误:如果 elevationRange 类型不是 Range<Double>
let invalidPath: KeyPath<Hike.Observation, Range<Double>> = \.invalidProperty
2. 动态访问属性

KeyPath 允许在不持有实例的情况下,通过类型信息操作属性。例如:

func getRange(from observation: Hike.Observation, via path: KeyPath<Hike.Observation, Range<Double>>) -> Range<Double> {return observation[keyPath: path]
}
3. 函数式编程与数据驱动

在 SwiftUI 中,KeyPath 常用于声明式 UI 的数据绑定,例如:

struct HikeView: View {var observation: Hike.Observationvar path: KeyPath<Hike.Observation, Range<Double>>var body: some View {Text("Range: \(observation[keyPath: path].lowerBound)...\(observation[keyPath: path].upperBound)")}
}

三、SwiftUI 中的典型应用场景

1. 动态 UI 组件配置

当需要根据不同的 KeyPath 动态选择显示的属性时,可以避免重复代码。例如,一个用于展示不同范围(海拔、心率等)的通用组件:

struct RangeView: View {var observation: Hike.Observationvar path: KeyPath<Hike.Observation, Range<Double>>var body: some View {HStack {Text("Min: \(observation[keyPath: path].lowerBound)")Text("Max: \(observation[keyPath: path].upperBound)")}}
}// 使用方式
RangeView(observation: observation, path: \.elevationRange)
RangeView(observation: observation, path: \.heartRateRange)
2. 与 @dynamicMemberLookup 结合

SwiftUI 的 @dynamicMemberLookup 特性允许通过 KeyPath 实现更灵活的 API 设计。例如,自定义绑定:

@dynamicMemberLookup
struct ObservationBinding {let observation: Hike.Observationsubscript<T>(dynamicMember path: KeyPath<Hike.Observation, T>) -> T {observation[keyPath: path]}
}// 使用方式
let binding = ObservationBinding(observation: observation)
let elevationRange = binding.elevationRange // 通过 KeyPath 动态访问
3. 动画与状态管理

在 SwiftUI 的动画系统中,KeyPath 可以指定要动画化的属性:

withAnimation(.easeInOut) {// 通过 KeyPath 指定要动画化的属性observedObject[keyPath: path] = newRange
}

四、对比其他技术

技术特点适用场景
KeyPath类型安全,编译时检查需要动态但类型安全的属性访问
字符串 KVC动态但类型不安全Objective-C 兼容或运行时动态性需求
闭包访问(Hike.Observation) -> Range<Double>需要复杂计算或自定义逻辑时

五、进阶技巧

1. 类型擦除的 KeyPath

当需要将不同根类型的 KeyPath 存入集合时,可以使用 AnyKeyPath

let paths: [AnyKeyPath] = [\Hike.Observation.elevationRange, \Hike.Weather.temperatureRange]
2. KeyPath 组合

通过自定义运算符组合 KeyPath(需谨慎使用):

func + <Root, Intermediate, Value>(lhs: KeyPath<Root, Intermediate>,rhs: KeyPath<Intermediate, Value>
) -> KeyPath<Root, Value> {lhs.appending(path: rhs)
}// 使用方式
let path = \Hike.Observation.sensor + \Sensor.currentValue

总结

KeyPath<Hike.Observation, Range<Double>> 是 Swift 类型系统的强大体现,它在 SwiftUI 中广泛用于:

  1. 构建数据驱动的声明式 UI
  2. 实现类型安全的动态属性访问
  3. 减少模板代码,提升代码复用性

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

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

相关文章

IntelliJ IDEA 2024.1.4版无Tomcat配置

IntelliJ IDEA 2024.1.4 (Ultimate Edition) 安装完成后&#xff0c;调试项目发现找不到Tomcat服务&#xff1a; 按照常规操作添加&#xff0c;发现服务插件中没有Tomcat。。。 解决方法 1、找到IDE设置窗口 2、点击Plugins按钮&#xff0c;进入插件窗口&#xff0c;搜索T…

【个人开发】deepseed+Llama-factory 本地数据多卡Lora微调

文章目录 1.背景2.微调方式2.1 关键环境版本信息2.2 步骤2.2.1 下载llama-factory2.2.2 准备数据集2.2.3 微调模式2.2.4 微调脚本 2.3 踩坑经验2.3.1 问题一&#xff1a;ValueError: Undefined dataset xxxx in dataset_info.json.2.3.2 问题二&#xff1a; ValueError: Target…

SEO短视频矩阵系统源码开发概述

一、功能特性 多账号、多平台一键授权管理&#xff1a;该系统支持抖音、快手、小红书、B站和视频号等平台的账户集成&#xff0c;实现统一管理。批量视频发布及定时发布功能&#xff1a;用户能够通过系统进行大规模视频的批量上传和设定具体发布时间。AI混剪技术生成原创内容&…

Linux 服务器部署deepseek

把手教你在linux服务器部署deepseek&#xff0c;打造专属自己的数据库知识库 正文开始 第一步&#xff1a;安装Ollama 打开官方网址&#xff1a;https://ollama.com/download/linux 下载Ollama linux版本 复制命令到linux操作系统执行 [rootpostgresql ~]# curl -fsSL http…

DeepSeek-VL2 环境配置与使用指南

DeepSeek-VL2 环境配置与使用指南 DeepSeek-VL2 是由 DeepSeek 公司开发的一种高性能视觉-语言模型&#xff08;VLM&#xff09;。它是 DeepSeek 系列多模态模型中的一个版本&#xff0c;专注于提升图像和文本之间的交互能力。 本文将详细介绍如何配置 DeepSeek-VL2 的运行环…

EasyRTC轻量级SDK:智能硬件音视频通信资源的高效利用方案

在智能硬件这片广袤天地里&#xff0c;每一份资源的精打细算都关乎产品的生死存亡。随着物联网技术的疾速演进&#xff0c;实时音视频通信功能已成为众多设备的标配。然而&#xff0c;硬件资源的捉襟见肘&#xff0c;让开发者们常常陷入两难境地。EasyRTC&#xff0c;以它的极致…

Github Action自动流翻译README文档【CI/CD】

翻译自述文件操作 一、自述文件翻译 英语简体中文繁体中文印地语法语阿拉伯 GitHub Action 将自述文件翻译成任何语言 这是一个 GitHub Action&#xff0c;可以自动将你的 repo 中的自述文件翻译成指定的语言。 二、设置 添加工作流文件到您的项目&#xff08;例如.githu…

张弛语言课退费动漫配音与人物的深度剖析退费

在动漫的奇幻世界里&#xff0c;精彩的画面固然吸睛&#xff0c;而配音更是赋予角色灵魂的关键要素&#xff0c;它与人物之间存在着千丝万缕的紧密联系。 《火影忍者》中的鸣人&#xff0c;他的配音充满活力与朝气&#xff0c;声音高亢且坚定&#xff0c;将鸣人的热血、乐观和…

Nginx负载均衡

一。Nginx负载均衡的算法以及过程 二。nginx四层负载均衡的配置&#xff08;四层&#xff09; 1.vi /etc/nginx/conf.d/lb.conf 比较常见&#xff1a;weight&#xff1a;设置权重&#xff0c;backup&#xff1a;当其他主机全部用不了&#xff0c;这个作为备份 2.systemctl r…

Python爬虫实战:股票分时数据抓取与存储 (1)

在金融数据分析中&#xff0c;股票分时数据是投资者和分析师的重要资源。它能够帮助我们了解股票在交易日内的价格波动情况&#xff0c;从而为交易决策提供依据。然而&#xff0c;获取这些数据往往需要借助专业的金融数据平台&#xff0c;其成本较高。幸运的是&#xff0c;通过…

json-schema 的编辑器

最近在找一个 json-schema 的编辑器&#xff0c;在网上找了找&#xff0c;以下两个项目用的比较多 一、两款json-schema-editor 1、vue-json-schema-editor-visual 一个高效易用的基于 Vue Element UI 的 json-schema 编辑器。 git地址&#xff1a;https://github.com/gis…

记一次Self XSS+CSRF组合利用

视频教程在我主页简介或专栏里 &#xff08;不懂都可以来问我 专栏找我哦&#xff09; 目录&#xff1a;  确认 XSS 漏洞 确认 CSRF 漏洞 这个漏洞是我在应用程序的订阅表单中发现的一个 XSS 漏洞&#xff0c;只能通过 POST 请求进行利用。通常情况下&#xff0c;基于 POST 的…

API网关基础知识总结

什么是网关&#xff1f; 微服务背景下&#xff0c;一个系统被拆分为多个服务&#xff0c;但是像安全认证&#xff0c;流量控制&#xff0c;日志&#xff0c;监控等功能是每个服务都需要的&#xff0c;没有网关的话&#xff0c;我们就需要在每个服务中单独实现&#xff0c;这使…

Redis存储⑥Redis五大数据类型之 Zset

目录 1. Zset 有序集合 1.1 Zset 有序集合常见命令 zadd zcard zcount zrange zrevrange zrangebyscore&#xff08;弃用&#xff09; zpopmax bzpopmax zpopmin bzpopmin zrank zrevrank zscore zrem zremrangebyrank zremrangebyscore zincrby 1.2 Zset有…

景联文科技:以精准标注赋能AI未来,打造高质量数据基石

在人工智能蓬勃发展的时代&#xff0c;数据已成为驱动技术革新的核心燃料&#xff0c;而高质量的数据标注则是让AI模型从“感知”走向“认知”的关键桥梁。作为深耕数据服务领域的创新者&#xff0c;景联文科技始终以“精准、高效、安全”为核心理念&#xff0c;为全球AI企业提…

Wireshark TS | 再谈虚假的 TCP Spurious Retransmission

前言 在之前的《虚假的 TCP Spurious Retransmission》文章中曾提到一个错误判断为 TCP Spurious Retransmission&#xff0c;实际为 TCP Out-Of-Order 的案例&#xff0c;本次继续探讨一个虚假的 TCP Spurious Retransmission 案例。 问题背景 TCP Spurious Retransmission…

Redis常用的五种数据结构详解

一、Redis 数据库介绍 Redis 是一种键值&#xff08;Key-Value&#xff09;数据库。相对于关系型数据库&#xff08;比如 MySQL&#xff09;&#xff0c;Redis 也被叫作非关系型数据库。 像 MySQL 这样的关系型数据库&#xff0c;表的结构比较复杂&#xff0c;会包含很多字段&…

Effective Objective-C 2.0 读书笔记——内存管理(上)

Effective Objective-C 2.0 读书笔记——内存管理&#xff08;上&#xff09; 文章目录 Effective Objective-C 2.0 读书笔记——内存管理&#xff08;上&#xff09;引用计数属性存取方法中的内存管理autorelease保留环 ARCARC必须遵循的方法命名原则ARC 的自动优化&#xff1…

PyTorch 混合精度训练中的警告处理与代码适配指南

在最近的 PyTorch 项目开发中&#xff0c;遇到了两个与混合精度训练相关的警告信息。这些警告主要涉及 torch.cuda.amp 模块的部分 API 已被标记为弃用&#xff08;deprecated&#xff09;。本文将详细介绍这些警告的原因、解决方法以及最佳实践。 警告内容 警告 1: torch.cud…

STM32自学记录(九)

STM32自学记录 文章目录 STM32自学记录前言一、DMA杂记二、实验1.学习视频2.复现代码 总结 前言 DMA 一、DMA杂记 DMA&#xff08;Direct Memory Access&#xff09;直接存储器存取 DMA可以提供外设和存储器或者存储器和存储器之间的高速数据传输&#xff0c;无须CPU干预&…