//
// PropertyClass.swift
// swift属性
//
// Created by wsy on 15/8/25.
// Copyright (c) 2015年 WSY. All rights reserved.
//
import UIKit
class PropertyClass: NSObject {
/**
* 属性是描述特定类、结构或者枚举的值。存储属性作为实例的一部分存储常量与变量的值,而计算属性计算他们的值(不只是存储)。计算属性存在于类、结构与枚举中。存储属性仅仅只在类与结构中。
属性通常与特定类型实例联系在一起。但属性也可以与类型本身联系在一起,这样的属性称之为类型属性。
*/
// 1. 存储属性:
/**
* 存储属性存储着常量或者变量的值。存储属性可分为变量存储属性(关键字var描述)和常量存储属性(关键字let描述)。
*/
struct FixLengteRange {
// 存储属性
var firstValue: Int
let length: Int
}
func userFixLengthRange(){
var rangeItem = FixLengteRange(firstValue: 0, length: 3)
rangeItem.firstValue = 4
// rangeItem.length = 3 error
println("\(rangeItem.firstValue)")
let rangItem2 = FixLengteRange(firstValue: 2, length: 4)
// 因为rangeOfFourItems是一个常量(let),即便firstValue是一个变量属性,它的值也是不可以被改变的
// rangItem2.firstValue = 2
}
// 2、计算属性
// 除了存储属性,类、结构和枚举能够定义计算属性。计算属性并不存储值,它提供getter和可选的setter来间接地获取和设置其它的属性和值。
struct MyPoint {
var x = 0.0, y = 0.0
}
struct Mysize {
var width = 0.0, height = 0.0
}
struct MyRect {
var orign = MyPoint()
var size = Mysize()
/**
* Rect结构包含一个称之为center的计算属性。Rect当前中心点的坐标可以通过origin和size属性得来,所以并不需要显式地存储中心点的值。取而代之的是,Rect定义一个称为center的计算属性,它包含一个get和一个set方法,通过它们来操作长方形的中心点,就像它是一个真正的存储属性一样。
*/
var center: MyPoint{
get{
let centerX = orign.x + (size.width/2)
let centerY = orign.y + (size.height/2)
return MyPoint(x: centerX, y: centerY)
}
set(newCenter){
orign.x = newCenter.x - (size.width / 2)
orign.y = newCenter.y - (size.height / 2)
}
// set 方法简写 默认地使用newValue这个名称来代替
// set{
// orign.x = newValue.x - size.width/2
// orign.y = newValue.y - size.height/2
// }
}
}
// 3.只读计算属性
/**
* 只读计算属性只带有一个getter方法,通过点操作符,可以放回属性值,但是不能修改它的值。
应该使用var关键字将计算属性-包含只读计算属性-定义成变量属性,因为它们的值并不是固定的。let关键字
只被常量属性说使用,以表明一旦被设置它们的值就是不可改变的了
通过移除get关键字和它的大括号,可以简化只读计算属性的定义:
这个例子定义了一个三维长方体结构Cuboid,包含了长宽高三个属性,和一个表示长方体容积的只读计算属性volume。volume值是不可被设置的,因为它直接由长宽高三个属性计算而来。通过提供这样一个只读计算属性,Cuboid使外部用户能够访问到其当前的容积值。
*/
struct Cuboid {
var width = 0.0, heigth = 0.0, depth = 0.0
var volume: Double{
return width*heigth*depth
}
}
var myCuboid = Cuboid(width: 2, heigth: 4, depth: 5)
// 4.类型属性
/**
* 在C和Objective-C中,定义静态常量、变量和全局静态变量一样。但是在swift中,类型属性的定义要放在类型定义中进行,在类型定义的大括号中,显示地声明它在类型中的作用域。
对值类型而言,定义类型属性使用static关键字,而定义类类型的类型属性使用class关键字。下面的例子展示了存储和计算类型属性的用法:
*/
struct SomeStrcucture {
static var storedTypeProperty = "Hello SomeStrcucture"
static var computedTypeProperty: Int{
get{
return 3
}
set{
}
}
}
enum SomeEnumeration{
static var storedTypeProperty = "Hello SomeEnumeration"
static var computedTypeProperty: Int{
get{
return 4
}
set{
}
}
}
}
// 2.Lazy Stored Properties(懒惰存储属性?)
// 懒惰存储属性是当它第一次被使用时才进行初值计算。通过在属性声明前加上@lazy来标识一个懒惰存储属性。
class DataImporter {
var fileName: String = "data.text"
}
class DataManager {
// DataManager实例被创建时,并不需要马上就创建一个新的DataImporter实例
// DataImporter的实例importer只有在当它在第一次被访问时才被创建。例如它的fileName属性需要被访问时:
// println("\(manager.imaporter.fileName)")
lazy var imaporter = DataImporter()
// 存储属性
var data = [String]()
}
// 4、属性观察者
/**
* 属性观察者观察属性值的改变并对此做出响应。当设置属性的值时,属性观察者就被调用,即使当新值同原值相同时也会被调用。
除了懒惰存储属性,你可以为任何存储属性加上属性观察者定义。另外,通过重写子类属性,也可以继承属性(存储或计算)加上属性观察者定义。属性重写在“重写”章节定义。
注意
不必为未重写的计算属性定义属性观察者,因为可以通过它的setter方法直接对值的改变做出响应
定义属性的观察者时,你可以单独或同时使用下面的方法:
willSet:设置值前被调用
didSet:设置值后立刻被调用
当实现willSet观察者时,新的属性值作为常量参数被传递。你可以为这个参数起一个名字,如果不的话,这个参数就默认地被命名成newValue。
在实现didSet观察者时也是一样,只不过传递的产量参数表示的是旧的属性值。
注意:
属性初始化时,willset和didSet并不会被调用。只有在初始化上下文之外,当设置属性值时才被调用
下面是一个willSet和didSet用法的实例。定义了一个类StepCounter,用来统计人走路时的步数。它可以从计步器或其它计数器上获取输入数据,对日常联系锻炼的步数进行追踪。
*/
class StepCounter {
var stepCount: Int = 0{
willSet{
// 新值
println("About to set totalSteps to \(newValue)")
}
didSet{
// 旧值
println("old value \(oldValue)")
}
}
}