• iOS开发Swift-属性和方法


    一、类的定义

    Swift与Objective-C定义类的区别

    Objective-C:一般需要2个文件,1个.h声明文件和1个.m实现文件

    Swift:只需要1个.swift文件

    Swift中类的定义格式

    1 class 类名  {
    2     // ... 属性和方法
    3 }

    二、属性

    1.什么是属性

      Swift中的属性(Properties),就类似于其他面向对象语言中的成员变量

    2.属性的分类

    按照官方文档的说明,属性可以分为以下几种

    (1)存储属性(Stored Properties)

    (2)计算属性(Computed Properties)

    (3)类型属性(Type Properties)

    3.存储属性

    1)简单说明

    存储属性:存储属性就是存储在对象(实例)中的一个变量或者常量

    存储属性类似于其他面向对象语言中的成员变量

    1 class Person {
    2     var age: Int = 1
    3     var height: Double = 0.0
    4     let life = 1
    5 }

    说明:

    Person类中定义了3个存储属性

    2个变量存储属性:Int类型的age、Double类型的height 

    1个常量存储属性:Int类型的life

    系统并不会自动初始化上面的3个存储属性,需要进行手动初始化

    2)存储属性的读写

      如何读写存储属性?

      直接通过点运算符(.),就可以读写某个对象的存储属性  

    class Person {
        var age: Int = 1
        let life = 1
    }
    var p = Person()
    p.age = 20
    println("p的生命是(p.life), p的年龄是(p.age)")
    View Code

    说明:

    第5行:创建Person对象

    第6行:给对象p的age属性赋值

    第7行:访问对象p的life属性值和age属性值

    3)延迟存储属性

    什么是延迟存储属性?

    延迟存储属性是第一次使用时才进行初始化的属性

    使用@lazy来标识一个延迟存储属性

    1 class Person {
    2     @lazy var dog: Dog = Dog()
    3 }
    4 var p = Person()
    5 println(p.dog)

    说明:

      执行第5行代码时,才会去初始化dog属性,才真正创建了Dog对象

    延迟存储属性的使用注意:延迟存储属性必须是变量,不能是常量

    延迟存储属性的好处:让某些资源用到时再去加载,避免一些不必要的资源浪费

    4.计算属性

    (1)什么是计算属性

    跟存储属性不一样的是,计算属性不是直接存储值,而是提供get和set

    get:用来取值,封装取值的过程

    set:用来设值,封装设值的过程

    (2)代码示例:

    class Square {
        // 正方形的宽度
          var  Double = 0.0
        // 正方形的周长
          var girth: Double {
            get {
                // 周长 = 宽度 * 4
                    return width * 4
            }
            set(newGirth) {
                // 宽度 = 周长 / 4
                       width = newGirth / 4
            }
        }
    }
    View Code

    计算属性举例:

    1 var s = Square()
    2 s.width = 10
    3 println(s.girth)
    4 s.girth = 200 
    5 println(s.width) 

    说明:

    第3行代码:调用girth属性的get,输出结果是40

    第4行代码:调用girth属性的set,并且把200传递给newGirth参数

    第5行代码:输出结果是50

    (3)简便的set

      set也可以不特意指定新值的参数名,新值的默认参数名叫做newValue

    var girth: Double {
        get {
            return width * 4
        }
        set {
            width = newValue / 4
        }
    }
    s.girth = 200
    View Code

    说明:

      执行第9行代码时:第6行中newValue的值就是200

    (4)计算属性的使用注意

      1)因为计算属性的值不是固定的,因此只能用var修饰计算属性,不能用let

    class Square {
        var girth: Double {
            get {
                return girth
            }
            set(newGirth) {
                girth = newGirth
            }
        }
    }
    View Code

    说明:上面的代码会引发死循环,第4行代码会引发循环调用get,第7行代码会引发循环调用set。

    2)一个属性不能既是存储属性,又是计算属性

    class Square {
        var girth: Double = 20.0 {
            get {
                return 10.0
            }
            set() {
                
            }
        }
    }
    View Code

    说明:上面的代码是错误的

    (5)只读计算属性

    什么是只读计算属性?只提供get,没有set的计算属性

    class Square {
        var  Double = 0.0
        var girth: Double {
            get {
                return width * 4
            }
        }
    }
    var s = Square()
    s.girth = 200
    View Code

    说明:第10行代码会报错

      只读计算属性的简写。只读计算属性可以省略get关键字

    class Square {
        var  Double = 0.0
        var girth: Double {
            return width * 4
        }
    }
    var s = Square()
    s.girth = 200
    View Code

    说明:第8行代码会报错

    5.类型属性

    (1)什么是类型属性?

    用class关键字修饰的属性,就是类型属性,也可以称为“类属性”

    class修饰的类型属性只能是计算属性,不能是存储属性

    代码示例:

    1 class Circle {
    2     class var PI : Double  {
    3         return 3.14
    4     }
    5 }

    说明:第2行定义的属性PI,就是一个类型属性

    (2)类型属性的特点

      一个类只会有一份,类的多个实例对象都共享这唯一的一份

      类型属性的使用:类型属性不依赖于对象而存在,因此用类名来访问

      println(Circle.PI)

    三、属性监视器

    1.什么是属性监视器?

    有时,需要在属性值被修改的时候做出响应,这种情况下就可以用属性监视器

    属性监视器,可以监视属性值的修改过程 

      计算属性可以直接在set中监听属性值的改变,而存储属性没有set

    可以为存储属性添加willSet和didSet两个属性监视器

    (1)willSet:

    在设置新的属性值之前调用

    会将新的属性值作为参数传入,参数名默认是newValue

    (2)didSet:

    在设置新的属性值之后调用

    会将旧的属性值作为参数传入,参数明默认是oldValue

    2.代码示例

    class Square {
        var  Double = 0.0 {
            willSet {
                println("willSet---(newValue)")
            }
            didSet {
                println("didSet---(oldValue)")
            }
        }
    }
    var s = Square()
    s.width = 10
    View Code

    打印结果:

    willSet---10.0

    didSet---0.0

    3.使用注意

    class Square {
        var  Double = 0.0 {
            willSet { }
            didSet { width = 20 }
        }
    }
    var s = Square()
    s.width = 10
    println(s.width)
    View Code

    代码说明:

    willSet和didSet在属性初始化过程中不会被调用,只会当属性的值在初始化之外的地方被设置时被调用

    第2行代码的初始化不会引发willSet和didSet的调用

    第8行代码的赋值会引发willSet和didSet的调用

    如果在didSet监视器里为属性赋值,这个值会替换之前设置的值

    第4行对width属性进行了再次赋值,覆盖了第8行赋的值,因此第9行的输出结果是20

    willSet、didSet和set、get不能共存

    四、方法简单说明

    跟其他面向对象语言一样,Swift中的方法可以分为2大类:

    (1)实例方法(Instance Methods)

      在OC中,实例方法以减号(-)开头

    (2)类型方法(Type Methods)

      在OC中,类型方法以加号(+)开头

    五、实例方法

    1.什么是实例方法?

    实例方法:就是只能用对象实例调用的方法,也可以称为“对象方法”

    实例方法的语法跟函数基本一样

    2.代码示例:

    class Dog {
        func run() {
            println("Dog--->run")
        }
    }
    var d = Dog()
    d.run()
    View Code

    说明:

    第2行定义了一个run方法

    第7行调用run方法:调用方法也跟调用函数很像,格式是“对象名.方法名(参数)”

    六、方法的参数

    (1)方法和函数的参数有点区别,默认情况下

    方法的第1个参数名仅仅是局部参数名

    方法的其他参数名(除第1个参数名以外)既是局部参数名,又是外部参数名

    class Calculator {
        func sum(num1: Int, num2: Int) -> Int {
            return num1 + num2
        }
    }
    var c = Calculator()
    c.sum(10, num2: 20)
    View Code

    说明:

    num1仅仅是局部参数名,num2既是局部参数名,又是外部参数名

    相当于func sum(num1: Int, #num2: Int) -> Int

    (2)可以在参数名前面加个下划线 _,去掉默认的外部参数名

    class Calculator {
        func sum(num1: Int, _ num2: Int) -> Int {
            return num1 + num2
        }
    }
    var c = Calculator()
    c.sum(10, 20)
    View Code

    说明:num1、num2都仅仅是局部参数名,不是外部参数名

    (3)也可以给第1个参数增加外部参数名

    class Calculator {
        func sum(#num1: Int, num2: Int) -> Int {
            return num1 + num2
        }
    }
    var c = Calculator()
    c.sum(num1: 10, num2: 20)
    View Code

    说明:num1、num2既是局部参数名,又是外部参数名

    七、类型方法

    1.什么是类型方法?

      被关键字class修饰的方法,也可以称为“类方法”

    1 class Calculator {
    2     class func sum(num1: Int, num2: Int) -> Int {
    3         return num1 + num2
    4     }
    5 }
    6 Calculator.sum(10, num2: 20)

    说明:

    第2行定义了一个类型方法

    类型方法的特点:直接用类调用类型方法,不能用对象调用类型方法

    第6行调用类型方法,格式是“类名.方法名(参数)”

    注意:类型方法和实例方法的方法名可以一样

    class Calculator {
        class func sum(num1: Int, num2: Int) -> Int {
            return num1 + num2
        }
        func sum(num1: Int, num2: Int) -> Int {
            return num1 + num2
        }
    }
    Calculator.sum(10, num2: 20)
    var c = Calculator()
    c.sum(10, num2: 20)
    View Code

    说明:

    第2行定义的类型方法和第5行定义的实例方法:方法名一样

    第9行使用类调用类型方法

    第11行使用对象调用实例方法

    八、self

    1.简单说明

    在每个方法内部,都有个隐含的属性self,它的作用跟Objective-C中self的用法基本一致

    self代表的是什么?谁调用这个方法,self就代表谁

    (1)在实例方法中:self代表调用方法的某个对象

    (2)在类型方法中:self代表调用方法的某个类

    2.代码示例

    class Person {
        class func run() {
            println("class func run")
        }
        func run() {
            println("func run")
        }
        class func test() {
            self.run();
        }
        func test() {
            self.run();
        }
    }
    var p = Person()
    p.test()
    Person.test()
    View Code

    程序的输出结果是:

    func run

    class func run

  • 相关阅读:
    用perfmon简单分析GDI+性能和代码的一点小改进
    从clr profiler的角度看string concat 和stringbuilder的性能差别
    CLR兴趣小组第一次活动总结
    C#3.0看起来比较酷的咚咚
    windbg入门及提高(广告贴)
    济南.NET技术俱乐部8月份活动预告
    WebService安全性的问题
    Basic Windbg 4.Out Of Memory的分析及诊断方法
    A question about C++ static method and C# static method
    内存泄露的小问题
  • 原文地址:https://www.cnblogs.com/asd5551680/p/4178659.html
Copyright © 2020-2023  润新知