Swift学习:属性

本篇将详细总结介绍Swift属性的用法;
属性是与特定的Swift类、结构体、枚举相关联的值;与其他语言相比,属性不再是被类所特有。

主要内容:
1.存储属性与计算属性
2.属性观察器
3.类型属性
4.全局变量与局部变量

一、存储属性与计算属性

从属性被定义的方式上看,Swift属性有存储属性和计算属性两种:
存储属性:存储在特定类或结构体实例里的一个常量(let)或变量(var),作为实例的一部分;

计算属性:计算属性不直接存储值,而是提供一个getter和一个可选的setter,来间接设置其他属性或变量值;
下面通过一段代码演示这两种属性的区别:

struct Square{
    //存储属性
    var width:Double
    
    //计算属性:通过一定计算方法得到的属性
    var area:Double{
        get{
            return width * width
        }
    }
}
let square = Square(width: 10.0)
print("正方形边长:\(square.width)")     //正方形边长:10.0
print("正方形面积:\(square.area)")      //正方形面积:100.0

总结存储属性和计算属性的用法还有如下几种情况:

1.1.常量结构体的存储属性

如果创建一个结构体的实例并且将其赋值给一个常量,则无法再修改该实例的任何属性(包括其中的变量属性)。这是因为结构体是值类型,值类型实例被声明为常量,其所有属性都成了常量;在这点上,类与结构体不同,这种情况下,类中的可变属性可以被修改。

let square1 = Square(width: 10.0)
//square1.width = 11.0      //报错
var square2 = Square(width:20.0)
square2.width = 21.0      //可以修改

//注:如果Squre是一个类,那么以上两种情况都可以通过

1.2.延迟存储属性

延迟存储属性:第一次被调用的时候才会计算其初始值的属性。在属性声明前使用lazy来表示一个延迟存储属性。
延迟属性作用:当属性的值依赖于在实例的构造过程结束后才会知道影响值的外部因素时,或者当获得属性的初始值需要复杂或大量计算时,可以只在需要的时候计算它。

class Number{
    //存储属性
    var startNum: Int!
    var endNum:Int!
    //计算属性
    var length :Int{
        return endNum - startNum + 1
    }
    
    //延迟属性:使用闭包计算出了延迟属性的值,此过程只执行一次
    lazy var sum: Int = {
        print("计算延迟属性。。。。")
        var tempNum = 0;
        for i in self.startNum...self.endNum{
            tempNum += i;
        }
        return tempNum
    }()
    
    //可失败的构造方法
    init?(startNum: Int , endNum:Int){
        if(startNum > endNum){
            return nil
        }
        self.startNum = startNum
        self.endNum = endNum
    }
}

let number = Number(startNum: 1, endNum: 100)
number?.length  //100
number?.sum    //5050
number?.sum    //5050

注意
1.必须将延迟存储属性声明成变量(使用var关键字),因为属性的初始值可能在实例构造完成之后才会得到。而常量属性在构造过程完成之前必须要有初始值,因此无法声明成延迟属性。
2.如果一个被标记为 lazy 的属性在没有初始化时就同时被多个线程访问,则无法保证该属性只会被初始化一次。

1.3.计算属性的使用

计算属性不直接存储值,而是提供一个getter和一个可选的setter,来间接设置其他属性或变量值;总结它的使用特点如下:
1.只有getter没有setter的计算属性就是只读计算属性。只读属性通过点运算符访问,只能返回值而不可设置新值;
2.计算属性与其他属性相关,是变化的,所以必须使用var关键字进行修饰,包括只读计算属性;
3.只读计算属性可以去掉get关键字和花括号;

struct Point {
    var x = 0.0
    var y = 0.0
}
struct Size {
    var width = 0.0
    var height = 0.0
}
class Rectangle{
    //存储型数据
    var originPoint  = Point()
    var size = Size()
    
    //计算型属性
    var center:Point{
        //get方法:获取计算属性值
        get{
            let center_x = originPoint.x + size.width/2
            let center_y = originPoint.y + size.height/2
            return Point(x: center_x, y: center_y)
        }
        
        //如果没有set方法,是只读,
        /*
         set(newCenter){
            originPoint.x = newCenter.x - size.width/2
            originPoint.y = newCenter.y - size.height/2
         }
         */
        
        //set方法:设置计算属性新值
        //这里也可以省略括号和newCenter.使用newValue
        set{
            originPoint.x = newValue.x - size.width/2
            originPoint.y = newValue.y - size.height/2
        }
    }
    
    //计算属性:area属性只有get,可以不显式的声明出get;此属性为只读属性
    var area:Double{
        return size.width * size.height
    }
    
    init(origin: Point, size: Size){
        self.originPoint  = origin
        self.size  = size
    }
}
//创建一个长方形
var rect = Rectangle(origin: Point(x: 0, y: 0), size: Size(width: 100, height: 100))
rect.center
rect.area  //10000

二、属性观察器

属性观察器监控和响应属性值的变化,每次属性被设置值的时候都会调用属性观察器,即使新值和当前值相同的时候也不例外。

属性观察器可以为延迟属性外的其他存储属性添加属性观察,也可以通过继承的方式重写父类属性,为其添加属性观察期。但是我们没有必要为非重写的计算属性添加属性观察器,因为它本身就可以通过自己的setter直接监控和响应值的变化。

添加属性观察器方式如下:
willSet方法:
在新的值被设置之前调用,拥有一个默认参数newValue(代表新的属性值);
didSet方法:
在新的值被设置之后立刻调用,拥有一个默认参数oldValue(代表就的属性值);

下面通过lightBlub演示用法,其中为currentDianYa属性添加了观察器:

class lightBlub {
    //最大电压和当前电压
    static let maxDianYa = 30
    
    //属性监听
    //注意:willSet和didSet括号中的值可以省略,直接使用系统自带的newVlaue和oldValue
    var currentDianYa = 0 {
        //可以使用系统默认的属性newValue和oldValue
        willSet(newCurrentDianya){
            print("当前电压值将要改变:  \(currentDianYa) -> \(newCurrentDianya)")
        }
        
        //当调用此方法时,已经设置了值的时候,
        didSet(oldCurentDianYa){
            if(currentDianYa == lightBlub.maxDianYa){
                print("请注意 ,当前电压达到了最大电压值")
            }else if(self.currentDianYa > lightBlub.maxDianYa){
                print("当前电压过高,不能设置新的电压值")
                currentDianYa = oldCurentDianYa
            }
        }
    }
}
let light = lightBlub()
light.currentDianYa = 10
light.currentDianYa = 30
light.currentDianYa = 40
/*
 打印结果:
 当前电压值将要改变:  0 -> 10
 当前电压值将要改变:  10 -> 30
 请注意 ,当前电压达到了最大电压值
 当前电压值将要改变:  30 -> 40
 当前电压过高,不能设置新的电压值
*/

注意:willSet和didSet并不会在初始化时被调用

三、类型属性

实例属性属于一个特定类型的实例,因此实例之间的属性相互独立。但其实,也可以为类型本身定义属性,这样无论创建了多少个该类型实例,这些属性只有唯一的一份,这种属性就是类型属性

Swift的类型属性就相当于OC或者C中的类变量,但他们有着以下的不同:
在OC或者C中,与某个类型相关的静态常量和静态变量,是作为全局静态变量来定义的。但是Swift中,类型属性是作为类型定义的一部分写在类型最外层的花括号内,因此它的作用范围也就在类型支持的范围内。

Swift类型属性使用关键字static,下面是一个具体示例:

//测试Int的类型属性
Int.min 
Int.max

class Player {
    var name: String = ""
    //对象属性:本人的得分
    var score: Int = 0
    //类型属性:本游戏的最高得分,使用类名来访问,使用关键字static声明
    static var heighestScore:Int = 0;
    
    //构造方法
    init(name: String){
        self.name = name
    }
    //玩一句游戏得分
    func playGame(){
        let tempNum = Int(arc4random()%100)+1
        self.score += tempNum
        print("\(name) 的游戏得分是:\(score)")
        
        if(self.score > Player.heighestScore){
            Player.heighestScore = self.score
        }
        print("当前本游戏的最高分是:\(Player.heighestScore)")
    }
}
let player1 = Player(name: "zs")
player1.playGame()
player1.playGame()
let player2 = Player(name: "cf")
player2.playGame()
/*
 打印结果
 zs 的游戏得分是:11
 当前本游戏的最高分是:11
 zs 的游戏得分是:87
 当前本游戏的最高分是:87
 
 cf 的游戏得分是:88
 当前本游戏的最高分是:88
*/

四、全局变量与局部变量

全局变量:在函数、方法、闭包或者任意类型之外定义的变量
局部变量:在函数、方法或者闭包内部定义的变量

全局的常量或变量都是延迟计算的,跟延迟存储属性相似,不同的地方在于:全局的常量或变量不需要标记lazy修饰符。

局部范围的常量或变量从不延迟计算。

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

推荐阅读更多精彩内容