你是否已经用Swift开发了几个月,现在,你想成为一个更好的Swift开发者?少年,你来对了地方,我这里有一本失传多年的武林秘籍传授于你。不要在意代码的格式,我想尽我所能保持代码的简洁。以至于可以是你方便的拷贝到playground来进行代码验证
。
废话不多说,让我们愉快的开始一段Swift体验之旅吧。
1、Extension
求平方
普通版本
123func square(x: Int) -> Int { return x * x }var squaredOFFive = square(x: 5)square(x:squaredOFFive) // 625进阶版本
12345extension Int {var squared: Int { return self * self }}5.squared // 255.squared.squared // 625
总结
- 普通版本如果想要对同一个数进行多次求平方,还需要创建多余的局部变量,而且代码比较冗余。
- 进阶版本通过给
Int
添加一个extension,然后在extension中给Int
添加一个计算属性
。这样无需创建多余的局部变量即可获取平方值。关于计算属性相关知识可以点此了解
2、Generics
普通版本
123456var stringArray = ["Bob", "Bobby", "SangJoon"]var intArray = [1, 3, 4, 5, 6]var doubleArray = [1.0, 2.0, 3.0]func printStringArray(a: [String]) { for s in a { print(s) } }func printIntArray(a: [Int]) { for i in a { print(i) } }func printDoubleArray(a: [Double]) {for d in a { print(d) } }进阶版本
12func printElementFromArray<T>(a: [T]) {for element in a { print(element) } }
总结
- 普通版本:每个类型都要定义一个function,但是每个function的功能都是一样的,这样就造成了代码冗余。
- 进阶版本:只定义一个function,每个类型都可以使用。
3、For Loop vs While Loop
输出“Count”5次
普通版本
1234var i = 0while 5 > i {print("Count")i += 1 }进阶版本
1for _ in 1...5 { print("Count") }
总结
- 当我们不需要使用循环中的变量时,我们可以使用
For Loop
和_
来代替While Loop
。 - 记住:变量越多->问题越多->bug越多。
4、Optional Unwrapping
Gaurd let vs if let(新用户登录逻辑)
普通版本
12345678910var myUsername: Double?var myPassword: Double?// Hideous Codefunc userLogIn() {if let username = myUsername {if let password = myPassword {print("Welcome, \(username)"!)}}}进阶版本
1234func userLogIn() {guard let username = myUsername, let password = myPasswordelse { return }print("Welcome, \(username)!") }
译者注:1234//上述的普通版本也可写成这种形式if let username = myUsername, let password = myPassword { print("Welcome, \(username)"!) }
总结
- 普通版本代码结构嵌套冗余,进阶版本更加一目了然。
- 在判断一些前提必要条件的时候,推荐使用
guard let
,详情参见文档
5、Computed Property vs Function
计算直径
普通版本
12345678910111213// 💩 Codefunc getDiameter(radius: Double) -> Double { return radius * 2}func getRadius(diameter: Double) -> Double { return diameter / 2}getDiameter(radius: 10) // return 20getRadius(diameter: 200) // return 100getRadius(diameter: 600) // return 300//译者注var radi = 10.0var dia = getDiameter(radius: radi) // 20dia = 30.0 // 30radi // 10.0 此处必须调用getRadius函数radius的值才会改变进阶版本
123456789101112// Good Codevar radius: Double = 10var diameter: Double {get { return radius * 2}set { radius = newValue / 2}}radius // 10diameter // 20diameter = 1000radius // 500
总结
- 普通版本:创建了两个互转函数,代码冗余且没有对radius和diameter建立内在联系。当你修改diameter的时候,相应的radius并没有跟着修改。
- 进阶版本:通过计算属性实现两者内在逻辑关联,当diameter改变的时候,radius会跟随相应改变。
6、Enum to Type Safe
买票
普通版本
123456switch person {case "Adult": print("Pay $7")case "Child": print("Pay $3")case "Senior": print("Pay $4")default: print("You alive, bruh?")}进阶版本
1234567enum People { case adult, child, senior }var person = People.adultswitch person {case .adult: print("Pay $7")case .child: print("Pay $3")case .senior: print("Pay $4")}
总结
- 普通版本:通过硬编码的方式来判断字符串,不易维护且代码可读性不高。很可能漏掉某个条件而造成bug。
- 进阶版本:创建
enum
来实现所有条件,当你在switch中漏掉某个case的时候,编译器会提示错误,强制你判断所有case。这样可以减少bug,使代码更健壮。
7、Nil Coalescing
选择颜色
普通版本
123456var userChosenColor: String?var defaultColor = "Red"var colorToUse = ""if let Color = userChosenColor { colorToUse = Color } else{ colorToUse = defaultColor }进阶版本
1var colorToUse = userChosenColor ?? defaultColor
总结
- 使用
??
设置可选类型变量的默认值,减少冗余代码。
8、Conditional Coalescing
计算height
普通版本
1234567// Simply Verbosevar currentHeight = 185var hasSpikyHair = truevar finalHeight = 0if hasSpikyHair { finalHeight = currentHeight + 5}else { finalHeight = currentHeight }进阶版本
12// Lovely CodefinalHeight = currentHeight + (hasSpikyHair ? 5: 0)
总结
- 通过三目运算符来减少代码量,使逻辑更加清晰。
9、Functional Programming
获取偶数值
普通版本
12345678// Imperative (a.k.a boring)var newEvens = [Int]()for i in 1...10 {if i % 2 == 0 { newEvens.append(i) }}print(newEvens) // [2, 4, 6, 8, 10]进阶版本
12var evens = Array(1...10).filter { $0 % 2 == 0 }print(evens) // [2, 4, 6, 8, 10]
总结
- 使用
filter
替代上面的For-Loop
。这里也推荐大家更多的使用高阶函数。 - 更多高阶函数的使用请参见 Transforming an Array section。
10、Closure vs Func
普通版本
123// Normal Functionfunc sum(x: Int, y: Int) -> Int { return x + y }var result = sum(x: 5, y: 6) // 11进阶版本
123// Closurevar sumUsingClosure: (Int, Int) -> (Int) = { $0 + $1 }sumUsingClosure(5, 6) // 11
最后
希望这篇文章能给大家带来一点收获,让大家对Swift有更深的了解。
最后祝福大家五一快乐!