Swift学习笔记(五)--属性和方法

属性(Stored Properties)

类,结构体和枚举都可以拥有属性, 之前在枚举中已经稍微提过,属性分为存储属性和运算属性, 存储属性只能被类和结构体拥有.
和ObjC一样, 有实例变量也有类型变量, 同时, 我们也可以监听属性的变化.

  1. 存储属性
    和基础篇介绍的一样, 属性要声明为变量则用var, 否则用let, 只是相对于ObjC来说, 那些个nonautomatic, strong, weak等等怎么在Swift中对应起来?
    strong: 对引用类型默认的内存管理方式
    weak: 需要在var前加上weak来声明
    readOnly和readWrite是根据var还是let来确定的
    copy通过@NSCopying来声明
    之前也提过, Array, Dictionary和Set不是引用类型, 所以是以copy的形式赋值的

除此之外, Swift还引入了lazy关键字, 让这个属性在第一次用到的时候才初始化. 注意, lazy的属性必须是变量, 因为它一开始是不会被初始化出来的, 而常量是一开始必须要初始化的且后面不能更改.

  1. 运算属性
    之前提过, 运算属性就是并没有真正的实体, 是每次都需要运算得出的, 比如:
struct Point {
    var x = 0.0, y = 0.0
}
struct Size {
    var width = 0.0, height = 0.0
}
struct Rect {
    var origin = Point()
    var size = Size()
    var center: Point {
        get {
            let centerX = origin.x + (size.width / 2)
            let centerY = origin.y + (size.height / 2)
            return Point(x: centerX, y: centerY)
        }
        set(newCenter) { // 如果不写newCenter就会被用newValue的默认名
            origin.x = newCenter.x - (size.width / 2)
            origin.y = newCenter.y - (size.height / 2)
        }
    }
}
var square = Rect(origin: Point(x: 0.0, y: 0.0),
    size: Size(width: 10.0, height: 10.0))
let initialSquareCenter = square.center
square.center = Point(x: 15.0, y: 15.0)
print("square.origin is now at (\(square.origin.x), \(square.origin.y))")

对于Rect这个结构体, 有一个center的属性, 可见每次get的时候, 都需要用origin和size来运算得出, set的时候都需要运算一遍得到origin然后设置origin来体现的.(之前不写get也不写set就会当做get, 一般用作readOnly的属性, 如之前的description)

  1. 属性监听
    属性监听可以在存储属性发生变化的时候通知你, 我们可以为除声明为lazy之外的存储属性加监听. 也可以继承而来的属性(无论是运算还是存储)通过重载来加为之加监听(讲重载的时候会细讲). 需要注意的是不需要为非重写的计算属性添加属性观察器,因为可以通过它的 setter 直接监控和响应值的变化。
    可以用willSet和didSet来为属性定义监听, 如名字所示, 一个在设置值之前,一个在之后. 相关语法如下:
class StepCounter {
    var totalSteps: Int = 0 {
        willSet(newTotalSteps) { // 不写就是newValue
            print("About to set totalSteps to \(newTotalSteps)")
        }
        didSet {  // 默认是oldValue
            if totalSteps > oldValue  {
                print("Added \(totalSteps - oldValue) steps")
            }
        }
    }
}
let stepCounter = StepCounter()
stepCounter.totalSteps = 200
// About to set totalSteps to 200
// Added 200 steps
stepCounter.totalSteps = 360
// About to set totalSteps to 360
// Added 160 steps
stepCounter.totalSteps = 896
// About to set totalSteps to 896
// Added 536 steps

注意, 如果在一个属性的didSet观察器里为它赋值,这个值会替换之前设置的值。可以用做一些保护的值的设置.

  1. 全局和局部变量
    和其它语言差不多, 变量定义的在哪里确定它是全局变量还是局部变量. 根据官方定义则是, 声明在函数,方法或者闭包里面的就是局部变量, 否则就是全局变量. 需要注意的是, 全部变量都是lazy的方式初始化的.
    值得注意的是, 全局变量也可以是运算变量和加监听者, 但是官网并没有给出例子. 不过我估计差不多是这样的:
  var origin: CGPoint = CGPointMake(0, 0)
  var size: CGSize = CGSizeMake(100, 100)
  var center: CGPoint {
      get {
        return CGPoint(x: origin.x+size.width/2, y: origin.y+size.height/2)
      }
      set {
        origin.x = newValue.x - size.width/2
        origin.y = newValue.y - size.height/2
      }
  }

  center    // playground 右边显示 (x 50 y 50)
  center = CGPointMake(100, 100)
  origin   // playground 右边显示 (x 50 y 50)
  1. 类属性
    与ObjC差不多, 只不过需要用static来修饰. 类属性不需要初始化, 而且是默认带lazy的. 具体语法可以看:
struct SomeStructure {
    static var storedTypeProperty = "Some value."
    static var computedTypeProperty: Int {
        return 1
    }
}
enum SomeEnumeration {
    static var storedTypeProperty = "Some value."
    static var computedTypeProperty: Int {
        return 6
    }
}
class SomeClass {
    static var storedTypeProperty = "Some value."
    static var computedTypeProperty: Int {
        return 27
    }
    class var overrideableComputedTypeProperty: Int {
        return 107
    }
}

类属性的存取与实例属性差不多, 只不过是直接通过类来存取, 例如:

print(SomeStructure.storedTypeProperty)
// prints "Some value."
SomeStructure.storedTypeProperty = "Another value."
print(SomeStructure.storedTypeProperty)
// prints "Another value."
print(SomeEnumeration.computedTypeProperty)
// prints "6"
print(SomeClass.computedTypeProperty)
// prints "27"

属性这一节没有讲太多的新东西, 很多东西都是相似的,只是转换一下语法, 具体细节参考官方文档

方法

方法整个官方文档看了一遍, 感觉真的没有什么好讲的(普通实例方法和类方法, 不包含初始化方法和析构方法).
讲几点需要注意的吧:

  1. 结构体和枚举在方法中如果要修改属性, 则要加上mutating来修饰, 例如:
  struct Point {
      var x = 0.0, y = 0.0
      mutating func moveByX(deltaX: Double, y deltaY: Double) {
        x += deltaX
        y += deltaY
      }
  }  

  enum TriStateSwitch {
    case Off, Low, High
    mutating func next() {
        switch self {
        case Off:
            self = Low
        case Low:
            self = High
        case High:
            self = Off
        }
      }
}
  1. 类方法和之前提类属性一样, 用static修饰, 例如:
struct LevelTracker {
    static var highestUnlockedLevel = 1
    static func unlockLevel(level: Int) {
        if level > highestUnlockedLevel { highestUnlockedLevel = level }
    }
    static func levelIsUnlocked(level: Int) -> Bool {
        return level <= highestUnlockedLevel
    }
    var currentLevel = 1
    mutating func advanceToLevel(level: Int) -> Bool {
        if LevelTracker.levelIsUnlocked(level) {
            currentLevel = level
            return true
        } else {
            return false
        }
    }
}
  1. 这里先提一下init方法和deinit方法, 之后会详细讲, init/deinit方法是不需要用func来声明的, 而且deinit是不能有参数的, 而且不要自己去调用, 如:
class Player {
    var tracker = LevelTracker()
    let playerName: String
    func completedLevel(level: Int) {
        LevelTracker.unlockLevel(level + 1)
        tracker.advanceToLevel(level + 1)
    }
    init(name: String) {
        playerName = name
    }
    deinit {
      
    }
}

实在没有太多新鲜的东西, 唯一有讲头的是init和deinit, 但是苹果把它们拆开成两个章节来讲了. 之后再深入讨论吧. 具体细节参考官方文档

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容