写在前面
接触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)!" ) } |