美文网首页Swift以及混编
Swift-类、结构体、属性、方法

Swift-类、结构体、属性、方法

作者: wg刚 | 来源:发表于2019-01-09 16:43 被阅读0次

    最近项目使用的是OC,后头看之前用Swift开发的一个项目时,发现很多细节都忘记了😭😭。
    为了回忆和以后方便查看,现在根据官方文档swift编程语言,做下笔记。

    1、 值类型和引用类型

    • 值类型被赋予给一个变量、常量或者被传递给一个函数的时候,其值会被拷贝。

    在 Swift 中,所有的基本类型:整数、浮点数、布尔值、字符串、数组和字典,都是值类型,并且在底层都是以结构体的形式所实现。

    所有的结构体和枚举类型都是值类型。这意味着它们的实例,以及实例中所包含的任何值类型属性,在代码中传递的时候都会被复制。

    • 引用类型在被赋予到一个变量、常量或者被传递到一个函数时,其值不会被拷贝,引用的是已存在的实例本身而不是其拷贝。

    类是引用类型,因此引用的是已存在的实例本身而不是其拷贝。

    2、OC与Swift中的字符串、数组、和字典类型

    Swift 中,String,Array和Dictionary类型均以结构体的形式实现。这意味着被赋值给新的常量或变量,或者被传入函数或方法中时,它们的值会被拷贝。

    Objective-C 中NSString,NSArray和NSDictionary类型均以类的形式实现,而并非结构体。它们在被赋值或者被传入函数或方法时,不会发生值拷贝,而是传递现有实例的引用。

    3、类和结构体的选择

    按照通用的准则,当符合一条或多条以下条件时,考虑构建结构体:

    • 该数据结构的主要目的是用来封装少量相关简单数据值。
    • 有理由预计该数据结构的实例在被赋值或传递时,封装的数据将会被拷贝而不是被引用。
    • 该数据结构中储存的值类型属性,也应该被拷贝,而不是被引用。
    • 该数据结构不需要去继承另一个既有类型的属性或者行为。

    以下情境中适合使用结构体:

    • 几何形状的大小,封装一个width属性和height属性,两者均为Double类型。
    • 一定范围内的路径,封装一个start属性和length属性,两者均为Int类型。
    • 三维坐标系内一点,封装x,y和z属性,三者均为Double类型。

    在所有其它案例中,定义一个类,生成一个它的实例,并通过引用来管理和传递。实际中,这意味着绝大部分的自定义数据构造都应该是类,而非结构体。

    4、类和结构体

    • 类和结构体有着类似的定义方式。我们通过关键字class和struct来分别表示类和结构体,并在一对大括号中定义它们的具体内容:
    class SomeClass {
        // 在这里定义类
    }
    struct SomeStructure {
        // 在这里定义结构体
    }
    
    以下是定义结构体和定义类的示例:

    其中name为可选类型,值为可选String,默认值为nil,意为没有name值

    struct Resolution {
        var width = 0
        var height = 0
    }
    class VideoMode {
        var resolution = Resolution()
        var interlaced = false
        var frameRate = 0.0
        var name: String?
    }
    
    • 类和结构体实例
    let someResolution = Resolution()
    let someVideoMode = VideoMode()
    
    • 结构体类型的成员逐一构造器

    所有结构体都有一个自动生成的成员逐一构造器,用于初始化新结构体实例中成员的属性。

    let vga = Resolution(width:640, height: 480)
    

    类实例没有默认的成员逐一构造器

    • 结构体是值类型
    let hd = Resolution(width: 1920, height: 1080)
    var cinema = hd
    cinema.width = 2048
    
    print("cinema is now \(cinema.width) pixels wide")
    // 打印 "cinema is now 2048 pixels wide"
    print("hd is still \(hd.width) pixels wide")
    // 打印 "hd is still 1920 pixels wide"
    
    • 类是引用类型
    let tenEighty = VideoMode()
    tenEighty.resolution = hd
    tenEighty.interlaced = true
    tenEighty.name = "1080i"
    tenEighty.frameRate = 25.0
    
    let alsoTenEighty = tenEighty
    alsoTenEighty.frameRate = 30.0
    
    print("The frameRate property of tenEighty is now \(tenEighty.frameRate)")
    // 打印 "The frameRate property of theEighty is now 30.0"
    

    需要注意的是tenEighty和alsoTenEighty被声明为常量而不是变量。然而你依然可以改变tenEighty.frameRate和alsoTenEighty.frameRate,因为tenEighty和alsoTenEighty这两个常量的值并未改变。它们并不“存储”这个VideoMode实例,而仅仅是对VideoMode实例的引用。所以,改变的是被引用的VideoMode的frameRate属性,而不是引用VideoMode的常量的值。

    5、属性

    属性将值跟特定的类、结构或枚举关联。存储属性存储常量或变量作为实例的一部分,而计算属性计算(不是存储)一个值。
    注意计算属性可以用于类、结构体和枚举,存储属性只能用于类和结构体。Swift 中的属性没有对应的实例变量

    • 存储属性

    存储属性就是存储在特定类或结构体实例里的常量或变量。

    例如:FixedLengthRange 的实例包含一个名为 firstValue 的变量存储属性和一个名为 length 的常量存储属性。

    struct FixedLengthRange {
        var firstValue: Int
        let length: Int
    }
    var rangeOfThreeItems = FixedLengthRange(firstValue: 0, length: 3)
    // 该区间表示整数0,1,2
    rangeOfThreeItems.firstValue = 6
    // 该区间现在表示整数6,7,8
    

    注意如果创建了一个结构体的实例并将其赋值给一个常量,则无法修改该实例的任何属性,即使有属性被声明为变量也不行:

    let rangeOfFourItems = FixedLengthRange(firstValue: 0, length: 4)
    // 该区间表示整数0,1,2,3
    rangeOfFourItems.firstValue = 6
    // 尽管 firstValue 是个变量属性,这里还是会报错
    

    原因:结构体属于值类型,当值类型的实例被声明为常量的时候,它的所有属性也就成了常量。

    • Lazy Stored Properties

    延迟存储属性是指当第一次被调用的时候才会计算其初始值的属性。在属性声明前使用 lazy 来标示一个延迟存储属性。

    必须将延迟存储属性声明成变量(使用 var 关键字),因为属性的初始值可能在实例构造完成之后才会得到。而常量属性在构造过程完成之前必须要有初始值,因此无法声明成延迟属性。

    注意如果一个被标记为 lazy 的属性在没有初始化时就同时被多个线程访问,则无法保证该属性只会被初始化一次。

    • 计算属性

    计算属性不直接存储值,而是提供一个 getter 和一个可选的 setter,来间接获取和设置其他属性或变量的值。

    注意必须使用 var 关键字定义计算属性,包括只读计算属性,因为它们的值不是固定的。let 关键字只用来声明常量属性,表示初始化后再也无法修改的值。
    例如:Rect提供了一个名为center 的计算属性。

    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) {
                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))")
    // 打印 "square.origin is now at (10.0, 10.0)”
    
    • 简化 setter 声明

    如果计算属性的 setter 没有定义表示新值的参数名,则可以使用默认名称 newValue。

    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 {
                origin.x = newValue.x - (size.width / 2)
                origin.y = newValue.y - (size.height / 2)
            }
        }
    
    • 只读计算属性

    只有 getter 没有 setter 的计算属性就是只读计算属性。

    只读计算属性的声明可以去掉 get 关键字和花括号:

    struct Cuboid {
        var width = 0.0, height = 0.0, depth = 0.0
        var volume: Double {
            return width * height * depth
        }
    }
    let fourByFiveByTwo = Cuboid(width: 4.0, height: 5.0, depth: 2.0)
    print("the volume of fourByFiveByTwo is \(fourByFiveByTwo.volume)")
    // 打印 "the volume of fourByFiveByTwo is 40.0"
    
    • 属性观察

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

    可以为除了延迟存储属性之外的其他存储属性添加属性观察器,也可以通过重写属性的方式为继承的属性(包括存储属性和计算属性)添加属性观察器。

    willSet 在新的值被设置之前调用
    didSet 在新的值被设置之后调用

    class StepCounter {
        var totalSteps: Int = 0 {
            willSet(newTotalSteps) {
                print("About to set totalSteps to \(newTotalSteps)")
            }
            didSet {
                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
    
    • 类型属性(Type Properties)

    存储型类型属性可以是变量或常量,计算型类型属性跟实例的计算型属性一样只能定义成变量属性。

    跟实例的存储型属性不同,必须给存储型类型属性指定默认值,因为类型本身没有构造器,也就无法在初始化过程中使用构造器给类型属性赋值。

    1、类型属性语法

    Swift 中,类型属性是作为类型定义的一部分写在类型最外层的花括号内,因此它的作用范围也就在类型支持的范围内。

    使用关键字 static 来定义类型属性。在为类定义计算型类型属性时,可以改用关键字 class 来支持子类对父类的实现进行重写。

    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
        }
    }
    
    2、类型属性是通过类型本身来访问,而不是通过实例。
    print(SomeStructure.storedTypeProperty)
    // 打印 "Some value."
    SomeStructure.storedTypeProperty = "Another value."
    print(SomeStructure.storedTypeProperty)
    // 打印 "Another value.”
    

    6、方法

    • 实例方法 (Instance Methods)

    在 Objective-C 中,类是唯一能定义方法的类型。但在 Swift 中,类/结构体/枚举上均可以定义方法。

    1、定义一个很简单的Counter类:
    class Counter {
        var count = 0
        func increment() {
            count += 1
        }
    }
    
    let counter = Counter()
    // 初始计数值是0
    counter.increment()
    // 计数值现在是1
    
    2、在实例方法中修改值类型

    结构体和枚举是值类型。默认情况下,值类型的属性不能在它的实例方法中被修改。
    但是:
    你可以为这个方法选择可变(mutating)行为,然后就可以从其方法内部改变它的属性;

    struct Point {
        var x = 0.0, y = 0.0
        mutating func moveByX(deltaX: Double, y deltaY: Double) {
            x += deltaX
            y += deltaY
        }
    }
    var somePoint = Point(x: 1.0, y: 1.0)
    somePoint.moveByX(2.0, y: 3.0)
    print("The point is now at (\(somePoint.x), \(somePoint.y))")
    // 打印 "The point is now at (3.0, 4.0)"
    

    注意不能在结构体类型的常量(a constant of structure type)上调用可变方法。

    let fixedPoint = Point(x: 3.0, y: 3.0)
    fixedPoint.moveByX(2.0, y: 3.0)
    // 这里将会报告一个错误
    
    3、在可变方法中给 self 赋值

    可变方法能够赋给隐含属性self一个全新的实例。上面Point的例子可以用下面的方式改写:

    struct Point {
        var x = 0.0, y = 0.0
        mutating func moveBy(x deltaX: Double, y deltaY: Double) {
            self = Point(x: x + deltaX, y: y + deltaY)
        }
    }
    

    新版的可变方法moveBy(x:y:)创建了一个新的结构体实例,它的 x 和 y 的值都被设定为目标值。调用这个版本的方法和调用上个版本的最终结果是一样的。

    • self 属性

    类型的每一个实例都有一个隐含属性叫做self,self完全等同于该实例本身。你可以在一个实例的实例方法中使用这个隐含的self属性来引用当前实例。

    上面例子中的increment方法还可以这样写:

    func increment() {
        self.count += 1
    }
    

    注意实例方法的某个参数名称与实例的某个属性名称相同的时候,参数名称享有优先权。这时你可以使用self属性来区分参数名称和属性名称。

    • 类方法

    定义在类型本身上调用的方法。
    在方法的func关键字之前加上关键字static,来指定类型方法。类还可以用关键字class来允许子类重写父类的方法实现。

    在 Objective-C 中,你只能为 Objective-C 的类类型(classes)定义类型方法(type-level methods)。
    在 Swift 中,你可以为所有的类、结构体和枚举定义类型方法。

    相关文章

      网友评论

        本文标题:Swift-类、结构体、属性、方法

        本文链接:https://www.haomeiwen.com/subject/anlgrqtx.html