写在前面
接触swift也有段时间了,通过这段时间自己接触和借鉴别人的经验,记录几点关于swift的小技巧,虽然不是什么高深的原理知识,但是在平常的项目中用到或许可以提高开发效率呢,哈哈,接下来就简单总结一下:
枚举(ENUM)
结合一个小场景, 比如我们要做一个关于性别判断的一个小case:
可能我们首先想到的就是这样,硬编码写入,每次都需要手动输入
1
2
3
4
5
6
7
8
9
10
|
var gender = ""
gender = "man"
switch gender {
case "man" :
print( "man" )
case "female" :
print( "female" )
default :
print( "other" )
}
|
但是如果我们利用枚举来写的话:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
enum Gender {
case man
case female
case other
}
var gType: Gender = .man
switch gType {
case .man:
print( "男性" )
case .female:
print( "女性" )
case .other:
print( "其他类型" )
default :
print( "我也不知道" )
}
|
把需要的类型写到enum 中,就会减少不小心输入错误的几率,而且不在enum范围中的实例,编辑器也会指出。
空合运算符
比如我们要定义某个label的默认text, 也就是说,如果我们没有去人为设置,这个label就会用默认text显示,我们可能会这么写:
1
2
3
4
5
6
7
8
|
var customText: String?
var defaultText = "您好"
var textToUse = ""
if let text = customText {
textToUse = text
} else {
textToUse = defaultText
}
|
如果换种简单的写法就成了这样:
1
2
3
4
|
var customText: String?
var defaultText = "您好"
var textToUse = ""
var textToUse = customText ?? defaultText
|
?? 的意思就是如果textToUse为nil则选择defaultText 否则选择customText
函数式
比如我们获取10以内的奇数, 第一想法是,用for循环呀:
1
2
3
4
5
6
7
|
var arr = [Int]()
for i in 1...10 {
if i % 2 == 1 {
arr.append(i)
}
}
print(arr)
|
当然了,必然可以计算出结果,如果换种思路呢,swift内置filter函数:
1
2
3
4
|
var arr = (1...10).filter { (num) -> Bool in
num % 2 == 1
}
print(arr)
|
闭包 \ 函数
举个栗子,两个字符串拼接
使用函数:
1
2
3
4
|
func sum(a: String, b: String) -> String {
return a + b
}
var result = sum(a: "你好" , b: "哈哈哈" )
|
如果使用闭包的话:
1
2
3
4
|
var sumStringClosure: (String, String) -> String = {
$0 + $1
}
sumStringClosure( "hello" , "world" )
|
有木有感觉easy了好多
convenience init 便利初始化
我们声明一个类,给这个类设置变量然后将其初始化
1
2
3
4
5
6
7
8
9
10
11
|
class Animal {
var dog: Int?
var cat: Int?
init(dog: Int, cat: Int) {
self.dog = dog
self.cat = dog
}
}
var daDi = Animal(dag: 2, cat: 4)
daDi.dog
daDi.cat
|
假如我们想在每次用到 Animal这个类的时候,我们想着给这个类中的dog和cat都设置好数量,我们就可以用convenience init来设置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
class Animal {
var dog: Int?
var cat: Int?
init(dog: Int, cat: Int) {
self.dog = dog
self.cat = dog
}
convenience init() {
self.init(dog: 10, cat: 10)
}
}
var daDi = Animal()
daDi.dog
daDi.cat
|
属性观察
自从swift更新到swift3之后 我们发现 变量的set\get方法发生改变了有木有,添加了一个很方便的属性就是willSet和didSet,比如我们如果求一个正方形的周长,使用function的话是这样子的 :
1
2
3
4
5
|
var length :Double?
func getDiameter(length: Double) -> Double {
return length * 4
}
getDiameter(length: 10)
|
如果我们使用变量的属性观察方法:
1
2
3
4
5
6
7
8
9
10
11
|
var perimeter: Double?
var length: Double? {
willSet {
print( "准备赋值中" )
}
didSet {
perimeter = length! * 4
}
}
length = 20
perimeter
|
willSet
是在属性发生改变之前调用的
didSet
是在属性发生改变之后调用的。
遍历方法
打印字符串:如果使用while 来
1
2
3
4
5
|
var i = 0
while i < 5 {
print( "irembeu" )
i += 1
}
|
我们必须要定义一个变量来确保打印达到我们要求的次数,但是我们定义越多的变量就意味着出错的风险越大,所以还是尽量少写点儿代码,改成for循环版:
1
2
3
|
for _ in 1...10 {
print( "itembeu" )
}
|
不需要我们定义多余变量,因为swift语法设计的时候用_来代替可以忽略的变量了
计算属性\ 函数
取圆的直径和半径:
1、使用函数:在这种情况下,我们根据半径求直径,根据已经直径求半径,需要写两个function
1
2
3
4
|
func getDiameter(radius: Double) -> Double { return radius * 2}
func getRadius(diameter: Double) -> Double { return diameter / 2}
getDiameter(radius: 20)
getRadius(diameter: 100)
|
2、使用变量的计算属性
1
2
3
4
5
6
7
8
9
10
11
12
13
|
var radius: Double = 10
var diameter: Double {
get {
return radius * 2
}
set {
radius = newValue / 2
}
}
radius // 20
diameter // 40
diameter = 600
radius // 300
|
我们知道直径和半径是相互依存的关系的,使用变量的计算属性这样看起来要比使用函数计算要简洁的多了。
泛型
如果我们需要将不同类型数组中的变量打印输出,我们可能会这样做:
1
2
3
4
5
6
|
var stringArr = [ "骑士" , "湖人" , "公牛" ]
var intArr = [1, 3, 4, 5, 6]
var doubleArr = [1.0, 2.0, 3.0]
func printStringArr(a: [String]) { for s in a { print(s) } }
func printIntArr(a: [Int]) { for i in a { print(i) } }
func printDoubleArr(a: [Double]) { for d in a { print(d) } }
|
我们需要定义不同类型的数组,如果类型多了,那我们做的无谓的工作就有点儿多了,这时候我们如果使用泛型来解决这个问题的话:
1
2
3
4
5
6
|
func printElementFromArr<T>(elements: [T]) {
for e in elements {
print(e)
}
}
printElementFromArr(elements: [1,2,3,4])
|
简单好多 有木有, 泛型是Swift语言强大的核心,泛型是对类型的抽象,使用泛型开发者可以更加灵活方便的表达代码意图。有参函数的参数必须有一个明确的参数类型,有些时候开发者会遇到这样一种情况,就像上面我们举的这个例子,由于变量有类型之分,实现相同的功能,可能需要重载成多个函数来实现,这大大浪费了开发成本,使用泛型,可以完美的解决这个问题。
拓展
swift 中没有了OC中的category 但是保留了extension, 我们和extension打交道的时候也就比较多了, 如果我们需要计算一个数的平方,我们可以声明一个函数, 像这样:
1
2
3
4
|
func squ(x: Int) -> Int {
return x * x
}
var s = squ(x: 10)
|
那如果我们要求10 的4次方的话,我们就要var s = squ(x: 10)
squ(x: s)创建多余变量。
使用拓展:
1
2
3
4
5
6
7
|
extension Int {
var squ: Int {
return self * self
}
}
10.squ
10.squ.squ
|
简洁,不需要多创建变量。
Gaurd let \ if let
检验用户名和密码:
1、使用if let, 我们需要一层层的嵌套
1
2
3
4
5
6
7
8
9
|
var uName: Double?
var uPassword: Double?
func userLogIn() {
if let username = uName {
if let password = uPassword {
print( "欢迎, \(username)" !)
}
}
}
|
2、使用gaurd let, 如果uName或者uPassword为nil的话,程序就会走return方法,提前结束运行, 否则就会print("欢迎, \(username)!")
1
2
3
4
5
6
7
|
var uName: Double?
var uPassword: Double?
func userLogIn() {
guard let username = uName, let password = uPassword
else { return }
print( "欢迎, \(username)!" )
}
|