Property Wrapper in Swift笔记

property wrapper(翻译为中文叫做属性包装器)可以为代码添加了一个层隔离,这层隔离是针对property的,添加的位置是:property的定义代码和property如何存储的代码之间

更简单通俗一点的描述是:

property的存和取的过程中添加了一些代码,这些代码逻辑就是Property Wrapper

Property Wrapper基本写法与工作原理

  • @propertyWrapper标记 + (class或struct或enum)来定义一个Property Wrapper
  • 该结构中必须声明一个wrappedValue属性,用于表示包装后的值
less 复制代码
@propertyWrapper
struct TwelveOrLess {
    private var number = 0
    var wrappedValue: Int {
        get { return number }
        set { number = min(newValue, 12) }
    }
}

struct SmallRectangle {
    @TwelveOrLess var height: Int
    @TwelveOrLess var width: Int
}

rectangle.height = 24
print(rectangle.height)
// Prints "12"

基本原理是,编译器会自动将Property Wrapper中的逻辑合成到每个应用的属性上,通过显式的Property Wrapper写法能够看出大致的工作原理

kotlin 复制代码
struct SmallRectangle {
    private var _height = TwelveOrLess()
    private var _width = TwelveOrLess()
    var height: Int {
        get { return _height.wrappedValue }
        set { _height.wrappedValue = newValue }
    }
    var width: Int {
        get { return _width.wrappedValue }
        set { _width.wrappedValue = newValue }
    }
}

Property Wrapper高级用法

现在我们再去审视Property Wrapper,可以更通俗易懂地描述一下它的工作原理:

在property的set和get过程中,引入一个中间层,这个中间层可以是class、struct或enum中的任意结构。在这个结构中可以加入任意逻辑

Property Wrapper的核心既然是这个中间层,那围绕中间层的定义,出现了一些Property Wrapper的高级用法

这些高级用法主要是来自Property Wrapper的不同初始化方法。那么Property Wrapper有几种初始化方式呢?

因为Property Wrapper可以用class、struct、enum任意类型表示,那初始化方法按照对应type写即可。但是,其中有一种初始化方法比较特殊---类似init(wrappedValue:),即初始化方法中包含wrappedValue参数的情况

所以,下面我们据此分成两类初始化方法进行描述

  • 系统默认或自定义初始化方法
  • wrappedValue参数的初始化方法

自定义初始化方法

kotlin 复制代码
@propertyWrapper
struct SmallNumber {
    private var maximum: Int
    private var number: Int


    var wrappedValue: Int {
        get { return number }
        set { number = min(newValue, maximum) }
    }

    init() {
        maximum = 12
        number = 0
    }
}

struct NarrowRectangle {
	@SmallNumber var number: Int
}
  • 上面代码中SmallNumber提供了一个初始化方法
  • NarrowRectanglenumber的写法,就是使用了上述初始化方法
  • 注意一点,由于使用了SmallNumber初始化方法,所以可以认为默认情况下,NarrowRectangle.number是被赋值为0的,所以let abc = NarrowRectangle()写法是不会编译报错的,因为number可以被正确初始化

wrappedValue参数的初始化方法

less 复制代码
@propertyWrapper
struct SmallNumber {
    private var maximum: Int
    private var number: Int

    var wrappedValue: Int {
        get { return number }
        set { number = min(newValue, maximum) }
    }


    init() {
        maximum = 12
        number = 0
    }
    init(wrappedValue: Int) {
        maximum = 12
        number = min(wrappedValue, maximum)
    }
    init(wrappedValue: Int, maximum: Int) {
        self.maximum = maximum
        number = min(wrappedValue, maximum)
    }
}

struct UnitRectangle {
    @SmallNumber var height: Int = 1
    @SmallNumber var width: Int = 2
    @SmallNumber(wrappedValue: 2, maximum: 5) var  height: Int // 3
    @SmallNumber(maximum: 9) var width: Int = 2 // 4
}
  • 参数中有wrappedValue参数时,为wrappedValue传参的方式比较特殊---此处@SmallNumber var height: Int = 1的写法就时将1作为wrappedValue进行传值
  • 原理上讲的话,就等价于UnitRectangle(height: SmallNumber(wrappedValue: 1), width: SmallNumber(wrappedValue: 2))
  • 还有更复杂的初始化,比如后面两种初始化方法
  • UnitRectangle中3和4写法则使用了第3个初始化方法

总结

  • Property Wrapper在property的存取过程中添加了一个中间层,可以增加自定义逻辑
  • Property Wrapper可以将重复代码进行抽离、复用
  • Property Wrapper的工作原理是编译器自动合成代码
  • Property Wrapper的工作原理要求property必须是var
  • 基于Property Wrapper原理可以应用于复杂的场景,比如简化Codable过程--参考CodableWrapperExCodable代码库
相关推荐
游戏开发爱好者85 小时前
日常开发与测试的 App 测试方法、查看设备状态、实时日志、应用数据
android·ios·小程序·https·uni-app·iphone·webview
黑码哥5 小时前
ViewHolder设计模式深度剖析:iOS开发者掌握Android列表性能优化的实战指南
android·ios·性能优化·跨平台开发·viewholder
2501_915106327 小时前
app 上架过程,安装包准备、证书与描述文件管理、安装测试、上传
android·ios·小程序·https·uni-app·iphone·webview
2501_915106327 小时前
使用 Sniffmaster TCP 抓包和 Wireshark 网络分析
网络协议·tcp/ip·ios·小程序·uni-app·wireshark·iphone
熊猫钓鱼>_>7 小时前
移动端开发技术选型报告:三足鼎立时代的开发者指南(2026年2月)
android·人工智能·ios·app·鸿蒙·cpu·移动端
徐同保1 天前
通过ip访问nginx的服务时,被第一个server重定向了,通过设置default_server解决这个问题
ios·iphone
2501_915918411 天前
在 iOS 环境下查看 App 详细信息与文件目录
android·ios·小程序·https·uni-app·iphone·webview
2501_916007471 天前
没有 Mac 用户如何上架 App Store,IPA生成、证书与描述文件管理、跨平台上传
android·macos·ios·小程序·uni-app·iphone·webview
夏幻灵2 天前
HTTPS全面解析:原理、加密机制与证书体
ios·iphone
TheNextByte12 天前
如何在iPhone上恢复已删除的笔记的综合指南
笔记·ios·iphone