Swift 十二讲 第七章 结构体和聚合体 ,访问控制
1. 结构体
Swift的Structure和类非常相似,同样都有属性方法函数等等。但有几个关键不同:
- 结构体无继承。所以自然也没有类下降之类的东西。
- 结构体是值拷贝。当你创建了一个结构体后,你的变量名的意思就是这个实例。而不是这个实例的名字。这和类是不同的。为了强调这点。我们看下面的例子:
// Value type example
struct S { var data: Int = -1 }
var a = S()
var b = a // a is copied to b
a.data = 42 // Changes a, not b
println("\(a.data), \(b.data)") // prints "42, -1"
// Reference type example
class C { var data: Int = -1 }
var x = C()
var y = x // x is copied to y
x.data = 42 // changes the instance referred to by x (and y)
println("\(x.data), \(y.data)") // prints "42, 42"
上面的例子取自
https://developer.apple.com/swift/blog/?id=10
算是少见的好例子。计算机书和网站上举例语法的时候,为了说明自己的例子可能在现实世界发生,往往会用什么电话号码本,计算器,员工纪录,算小费之类的愚蠢的例子。实际上那些例子非常分散读者的注意力。有时候几个a,b,c整数就可以清楚的说明语法的关键所在的问题,有的书弄好几段解释员工纪录是怎么回事。这不是故意害人么。笔者认为一个语言的feature,和这个feature在写app时候好不好用,有没有用,那是另一回事,应该尽全力和语言介绍分开。
- 结构体的变异方法
默认情况下,结构体的方法不能改变结构体的属性。但你可以用关键字mutating定义变异方法用来改变结构体的属性。例如:
struct a {
var kk = 1
mutating func b()
{
kk = 2
}
}
var d = a()
println(d.kk) //输出1
d.b()
println(d.kk) //变异方法被调用过了,所以输出为2
- 结构体的类型方法和类的相似,但是用static关键字。如下例:
struct a {
static func b()
{
println("结构体的类型方法被调用")
}
}
a.b() //输出为:"结构体的类型方法被调用"
2. 聚合体
聚合体就是一组名字的集合。然后给这组名字起了个名字。这个集合的名字,就是一个新的类型。
想想看, Int不就是所有整数集合的名字么?
例如:
enum ac {
case 兔
case 狗
case 熊猫
}
var a = ac.兔
switch a
{
case .兔 :
println("兔子")
default:
println()
} //输出:兔子
聚合体也有类型方法和实例方法。定义和用法和结构体类似。具体细节读者可以查手册,这里不多讲。
- 访问控制
Swift有三个访问控制关键字,访问范围从大到小为:
- public 这个关键字所修饰的项,可以在一切地方使用。
- internal 这是默认的访问状态,这个关键字修饰的项,只能在同模块内使用。
- private 这个关键字修饰的项,只能在同一个文件内被使用。
访问控制要注意两点:
-
和object c混合编程的时候,细节很麻烦。要小心注意。
-
Swift访问控制的大小匹配要符合常识。例如一个类是private的,你不能把这个类的实例用pubilc修饰。但反过来则是可以的。这是因为类比实例在逻辑上更靠前。一个类如果是private的,你也不能把它的属性用pubilc修饰。另外如果一个类或者类型的访问控制是public,其成员属性方法的默认访问控制将会是internal。也就是说,你需要给成员属性方法明确加上pubilc,其成员才能是public。
-
委托初始化
结构体都可以委托初始化。也就是说,它们可以有多个init函数,其中一个可以去调用另一个。这有什么好处呢? 这样可以在生成实例的时候更加灵活。例如你想有两种实例生成的办法,一种是传个参数让它生成你需要的实例。另一种是不传参数,让它生成默认的实例。这种用法类似于类的快捷初始化。
例如:
struct a {
init (c: Int){println(c)}
init(){
self.init(c:10)
}
}
var b = a(c:1) //输出1
b = a() //输出10
比较下面的类的快捷初始化:
class a {
init (c: Int){println(c)}
convenience init(){
self.init(c:10)
}
}
var b = a(c:1) //输出1
b = a() //输出10