类与结构
swift的结构体增添了很多类的功能: 定义函数,定义下标,有构造函数,可以扩展,还可以遵循协议来提供某种标准化的功能。
并且结构有一个默认的逐一构造函数(每个属性都有的初始化函数).
结构和枚举是值类型 类是引用类型
在Swift中,很多基本数据类型,例如String、Array和Dictionary都是用结构实现的。也就意味着他们被赋值给一个常量或者变量、或者被传入一个方法时是通过复制的方式实现的。
一,属性
swift有两种属性。1 存储属性 2 计算属性
存储属性就是平时我们理解的属性,这个计算属性意思是可以通过存储属性计算出来的。上代码一看便知:
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()
//这个center是通过已知属性计算出来的(计算属性只能声明成var)
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) {
//这里不能在调用setcenter ,会导致死循环
origin.x = newCenter.x - (size.width / 2)
origin.y = newCenter.y - (size.height / 2)
}
}
}
在上面代码中origin和size是存储属性,center是计算属性.他里面这个set方法是可以改变相关存储属性值得。当没有set方法的时候叫做 “只读计算属性”,这时候可以吧get{ }省略,只留get里面的东西就可以了如下。
struct Rect {
var origin = Point();
var size = Size();
var center:Point{
let centerX = origin.x + (size.width / 2);
let centerY = origin.y + (size.height / 2);
return Point(x: centerX, y: centerY);
}
}
懒惰属性
当属性第一次被使用的时候才被初始化,就是oc中经常用哪个懒加载.只不过不用我们在getter方法里面自己判断为空的时候再初始化,声明的时候前面加一个lazy就哦了
lazy var obj = SomeClass();
属性观察者
可以为每个属性添加一个willSet,didSet来监听属性值的变化.这个比较方便,不用像oc那样添加观察者.
var age : Int {
willSet{
//此时age还没有改变,swift添加了默认参数newValue表示新值
//这里如果再调用 set age 不会造成死循环
// age = 10
}
didSet{
//此时age已经改变,同样swift自动添加了参数oldValue表示之前的值
}
}
如果一个带有观察者的属性被被当做in-out参数使用的时,会引起观察者被调用.
类属性
类属性和其他语言一样是属于类的属性,可直接通过类名使用,swift中结构,枚举,类都可以定义类属性.语法是在前面加一个static关键字, 可以是var 和 let
在为类定义计算型类型属性时,可以改用关键字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来声明类属性,因为overrideableComputedTypeProperty是个计算属性,并且希望子类可以重写这个计算实现
class var overrideableComputedTypeProperty: Int {
return 107
}
}
通过闭包设置默认属性
注意闭包结尾的大括号后面接了一对空的小括号。这用来告诉 Swift 立即执行此闭包。如果你忽略了这对括
号,相当于将闭包本身作为值赋值给了属性,而不是将闭包的返回值赋值给属性。
还有闭包中不能使用实例的其他属性,因为此时实例还没有初始化文成,也不能用self。
class SomeClass {
let someProperty: SomeType = {
// 在这个闭包中给 someProperty 创建一个默认值
// someValue 必须和 SomeType 类型相同
return someValue
}()
}
二, 方法
swift的类,结构和枚举都可以定义实例方法和类方法,是的没错 枚举也可以定义方法.
特点
1, 结构和枚举都是值类型。默认情况下,值类型的属性在自己的实例方法内部是不能修改的。就是说枚举和结构中的方法在默认情况下不能修改自身的属性。如果想修改,要在方法前面添加mutating 关键字.
2, swift函数的参数有两个标签
外部标签是为了提高代码的可读性,swift和oc一样努力让方法调用看起来想一个句子般易读。
func someFunc(外部调用参数 函数局部变量名:Int){ 函数内容}
func someFunc(outParam inParam:Int){ }
//调用时候是这样的
someFunc(outParam: 5);
//这个outParam可以不加 那么默认用inParam
func someFunc( inParam:Int){ }
someFunc(inParam: 5)
//如果一个都不想用的话,函数这样声明
func someFunc(_ inParam:Int){}
someFunc(5)
*在方法内部甚至可以给self重新赋值
struct Point {
var x = 0.0, y = 0.0
mutating func moveBy(x deltaX: Double, y deltaY: Double) {
x += deltaX
y += deltaY
}
}
var somePoint = Point(x: 1.0, y: 1.0)
somePoint.moveBy(x: 2.0, y: 3.0)
print("The point is now at (\(somePoint.x), \(somePoint.y))")
//在方法内部甚至可以给self重新赋值
enum TriStateSwitch {
case off, low, high
mutating func next() {
switch self {
case .off:
self = .low
case .low:
self = .high
case .high:
self = .off
}
}
}
var ovenLight = TriStateSwitch.low
ovenLight.next()
同样可以使用static来定义类方法,class类型可以使用class关键字,以允许子类重写父类的实现。注意swift的子类可以 重写父类的类方法。
三, 下标
下标语法是提供了一种方式让我们可以通过中括号的方式( someObj[] )去调用一段代码,有些时候更加方便的表示一种数据结构。
类,枚举,结构 都可以使用下标语法。
struct Matrix {
let rows: Int, columns: Int
var grid: [Double]
init(rows: Int, columns: Int) {
self.rows = rows
self.columns = columns
grid = Array(repeat: 0.0, count: rows * columns)
}
func indexIsValid(row: Int, column: Int) -> Bool {
return row >= 0 && row < rows && column >= 0 && column < columns
}
subscript(row: Int, column: Int) -> Double {
get {
assert(indexIsValid(row: row, column: column), "Index out of range")
return grid[(row * columns) + column]
}
set {
assert(indexIsValid(row: row, column: column), "Index out of range")
grid[(row * columns) + column] = newValue
}
}
}
//创建一个矩阵
var matrix = Matrix(rows: 2, columns: 2)
//使用下标语法设置矩阵的值(在表示矩阵这种数据结构的时候下标语法会更加清晰易读)
matrix[0, 1] = 1.5
matrix[1, 0] = 3.2
下标语法是这样的:通过 subscript 关键字
这个看起来和计算属性有点像。只不过他不是属性,倒是可以看做一种方法.
subscript(param)->resultType{
get{
}
set{
//set时候会有个默认参数newValue 就是即将set的值
}
}