更新chapter4代码高亮
capter4各小节代码部分增加```swift```语法显示;更新SUMMARY内容,增加Chapter4/03-05节标题;统一标题格式,更新SUMMARY,capter2/23_Access Control.md为capter2/23_Access_Control.md
This commit is contained in:
@ -1,4 +1,4 @@
|
||||
> 翻译:[老码团队翻译组-Arya](http://weibo.com/littlekok/)
|
||||
> 翻译:[老码团队翻译组-Arya](http://weibo.com/littlekok/)
|
||||
> 校对:[老码团队翻译组-Oberyn](http://weibo.com/u/5241713117)
|
||||
|
||||
# Access Control 权限控制的黑与白
|
||||
@ -40,7 +40,7 @@ Private(私有级别)的权限最严格,它可以用来隐藏某些功能
|
||||
除了可以给整个声明设权限,Swift还允许大家在需要的时候,把某个属性(property)的取值权限比赋值权限设得更加开放。
|
||||
|
||||
#####举个例子:
|
||||
|
||||
```swift
|
||||
public class ListItem {
|
||||
|
||||
// ListItem这个类,有两个公开的属性
|
||||
@ -68,7 +68,7 @@ Private(私有级别)的权限最严格,它可以用来隐藏某些功能
|
||||
return false
|
||||
}
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
当我们使用Objective-C和Swift混合开发时,需要注意:
|
||||
|
||||
|
||||
@ -1,4 +1,4 @@
|
||||
> 翻译:[老码团队翻译组-Tyrion](http://weibo.com/u/5241713117)
|
||||
> 翻译:[老码团队翻译组-Tyrion](http://weibo.com/u/5241713117)
|
||||
> 校对:[老码团队翻译组-Oberyn](http://weibo.com/u/5241713117)
|
||||
|
||||
# 造个类型不是梦-白话Swift类型创建
|
||||
@ -12,7 +12,7 @@
|
||||
- [支持Bool类型判断](#condition-by-bool)
|
||||
- [支持兼容各们各派的类型](#support-all-type)
|
||||
- [完善OCBool的布尔基因体系](#make-up-type)
|
||||
|
||||
|
||||
小伙伴们,Swift中的Bool类型有着非常重要的语法功能,并支撑起了整个Swift体系中的逻辑判断体系,经过老码的研究和学习, Bool类型本身其实是对基础Boolean类型封装,小伙伴们可能咬着手指头问老码,怎么一会Bool类型,一会Boolean类型,其区别在于,前者是基于枚举的组合类型,而后者则是基本类型,只有两种true和false。
|
||||
|
||||
<a name="prefix_expressions"></a>
|
||||
@ -21,10 +21,12 @@
|
||||
来我们先看一下OCBool的定义。
|
||||
|
||||
#####代码示例如下:
|
||||
1. enum OCBool{
|
||||
2. case ocTrue
|
||||
3. case ocFalse
|
||||
4. }
|
||||
```swift
|
||||
enum OCBool{
|
||||
case ocTrue
|
||||
case ocFalse
|
||||
}
|
||||
```
|
||||
|
||||
#####注意:
|
||||
|
||||
@ -34,12 +36,13 @@
|
||||
<a name="imp-default"></a>
|
||||
####实现默认值
|
||||
行,我们给了一个漂亮的定义,不过按照传统语言的经验,Bool值默认情况下是假, 所以我们的OCBool也应该如此,我们使用类型扩展技术增加这个默认特性:
|
||||
|
||||
1. extension OCBool{
|
||||
2. init(){
|
||||
3. self =.ocFalse
|
||||
4. }
|
||||
5. }
|
||||
```swift
|
||||
extension OCBool{
|
||||
init(){
|
||||
self =.ocFalse
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
#####注意:
|
||||
- 代码中第1行:extension关键字,非常强大,小伙伴们可以通过此创造出许多好玩的东西,建议各位去Github上看一个名为“Swiftz”的项目,它将扩展用到了极致。
|
||||
@ -47,50 +50,55 @@
|
||||
现在我们可以使用如下方法使用这个Bool类型。
|
||||
|
||||
#####代码示例如下:
|
||||
|
||||
1. var result:OCBool = OCBool()
|
||||
2. var result1:OCBool = .ocTrue
|
||||
```swift
|
||||
var result:OCBool = OCBool()
|
||||
var result1:OCBool = .ocTrue
|
||||
```
|
||||
|
||||
<a name="init-by-bool"></a>
|
||||
####支持基本布尔型初始化
|
||||
正如上述代码所述,我们只能通过类型或者枚举项目赋值,这是组合类型的用法,但是编码的日子里,我们总是希望和true,false直接打交道,也就是说,我们希望这么做,
|
||||
代码示例如下:
|
||||
|
||||
1. var isSuccess:OCBool = true
|
||||
```swift
|
||||
var isSuccess:OCBool = true
|
||||
```
|
||||
|
||||
如果小伙伴们直接这么用,则会出现如下错误:
|
||||
|
||||
1. /Users/tyrion-OldCoder/Documents/Learning/BoolType/BoolType/main.swift:30:24: Type 'OCBool' does not conform to protocol 'BooleanLiteralConvertible'
|
||||
|
||||
```
|
||||
/Users/tyrion-OldCoder/Documents/Learning/BoolType/BoolType/main.swift:30:24: Type 'OCBool' does not conform to protocol 'BooleanLiteralConvertible'
|
||||
```
|
||||
编译器咆哮的原因是,我们的类型没有遵从“布尔字面量转换协议”,接下来修正这个问题,
|
||||
#####代码示例如下:
|
||||
|
||||
1. import Foundation
|
||||
2.
|
||||
3. println("Hello, World!")
|
||||
4.
|
||||
5. enum OCBool{
|
||||
6. case ocTrue
|
||||
7. case ocFalse
|
||||
8. }
|
||||
9.
|
||||
10.
|
||||
11. extension OCBool: BooleanLiteralConvertible{
|
||||
12. static func convertFromBooleanLiteral( value: Bool) ->OCBool{
|
||||
13. return value ? ocTrue : ocFalse
|
||||
14. }
|
||||
15. }
|
||||
16.
|
||||
17. var isSuccess:OCBool = true
|
||||
```swift
|
||||
import Foundation
|
||||
|
||||
println("Hello, World!")
|
||||
|
||||
enum OCBool{
|
||||
case ocTrue
|
||||
case ocFalse
|
||||
}
|
||||
|
||||
|
||||
extension OCBool: BooleanLiteralConvertible{
|
||||
static func convertFromBooleanLiteral( value: Bool) ->OCBool{
|
||||
return value ? ocTrue : ocFalse
|
||||
}
|
||||
}
|
||||
|
||||
var isSuccess:OCBool = true
|
||||
```
|
||||
|
||||
#####注意:
|
||||
- 代码中的第11行是重点,我的类型OCBool支持了BooleanLiteralConvertible协议,这个协到底是干什么的呢,小伙伴们在Xcode代码编辑器,按住Command键,然后点击第11行中的BooleanLiteralConvertible协议名,则会进入它的定义,
|
||||
#####其定义如下:
|
||||
|
||||
1. protocol BooleanLiteralConvertible {
|
||||
2. typealias BooleanLiteralType
|
||||
3. class func convertFromBooleanLiteral(value: BooleanLiteralType) -> Self
|
||||
4. }
|
||||
```swift
|
||||
protocol BooleanLiteralConvertible {
|
||||
typealias BooleanLiteralType
|
||||
class func convertFromBooleanLiteral(value: BooleanLiteralType) -> Self
|
||||
}
|
||||
```
|
||||
|
||||
- 这个定义中有个类方法convertFromBooleanLiteral,它的参数为BooleanLiteralType类型,也就是我传入的Bool类型, 且返回值为实现这个协议的类型本身,在我们的OCBool类型中,其返回值就是OCBool本身。经过这个定义,我们可以直接对OCBool类型直接进行布尔字面量初始化了。
|
||||
|
||||
@ -98,64 +106,66 @@
|
||||
####支持Bool类型判断
|
||||
小伙伴们不安分, 肯定想着我怎么用它实现逻辑判断,所以如果你这么写,
|
||||
#####代码示例如下:
|
||||
|
||||
1. var isSuccess:OCBool = true
|
||||
2.
|
||||
3. if isSuccess {
|
||||
4. println( "老码请你吃火锅!")
|
||||
5. }
|
||||
```swift
|
||||
var isSuccess:OCBool = true
|
||||
|
||||
if isSuccess {
|
||||
println( "老码请你吃火锅!")
|
||||
}
|
||||
```
|
||||
|
||||
你永远吃不到老码的火锅,因为这里编译器会咆哮:
|
||||
|
||||
1. /Users/tyrion-OldCoder/Documents/Learning/BoolType/BoolType/main.swift:27:4: Type 'OCBool' does not conform to protocol 'LogicValue'
|
||||
|
||||
```
|
||||
/Users/tyrion-OldCoder/Documents/Learning/BoolType/BoolType/main.swift:27:4: Type 'OCBool' does not conform to protocol 'LogicValue'
|
||||
```
|
||||
OCBool现在只能用bool类型初始化,而不能直接返回bool型,小火把们还记得在《老码说编程之白话Swift江湖》中,老码多次提到,妈妈再也不担心我们 if a = 1{}的写法了, 因为等号不支持值返回了, 所以在if判断是后面的条件必须有返回值,OCBool没有,所以编译器哭了。我们解决这个问题。
|
||||
|
||||
#####代码示例如下:
|
||||
```swift
|
||||
import Foundation
|
||||
|
||||
1. import Foundation
|
||||
2.
|
||||
3. println("Hello, World!")
|
||||
4.
|
||||
5. enum OCBool{
|
||||
6. case ocTrue
|
||||
7. case ocFalse
|
||||
8. }
|
||||
9.
|
||||
10.
|
||||
11. extension OCBool: BooleanLiteralConvertible{
|
||||
12. static func convertFromBooleanLiteral( value: Bool) ->OCBool{
|
||||
13. return value ? ocTrue : ocFalse
|
||||
14. }
|
||||
15. }
|
||||
16.
|
||||
17. extension OCBool: LogicValue{
|
||||
18. func getLogicValue() ->Bool {
|
||||
19. var boolValue: Bool{
|
||||
20. switch self{
|
||||
21. case .ocTrue:
|
||||
22. return true
|
||||
23. case .ocFalse:
|
||||
24. return false
|
||||
25. }
|
||||
26. }
|
||||
27. return boolValue
|
||||
28. }
|
||||
29. }
|
||||
30.
|
||||
31.
|
||||
32. var isSuccess:OCBool = true
|
||||
33.
|
||||
34. if isSuccess {
|
||||
35. println( "老码请你吃火锅!")
|
||||
36. }
|
||||
println("Hello, World!")
|
||||
|
||||
enum OCBool{
|
||||
case ocTrue
|
||||
case ocFalse
|
||||
}
|
||||
|
||||
|
||||
extension OCBool: BooleanLiteralConvertible{
|
||||
static func convertFromBooleanLiteral( value: Bool) ->OCBool{
|
||||
return value ? ocTrue : ocFalse
|
||||
}
|
||||
}
|
||||
|
||||
extension OCBool: LogicValue{
|
||||
func getLogicValue() ->Bool {
|
||||
var boolValue: Bool{
|
||||
switch self{
|
||||
case .ocTrue:
|
||||
return true
|
||||
case .ocFalse:
|
||||
return false
|
||||
}
|
||||
}
|
||||
return boolValue
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
var isSuccess:OCBool = true
|
||||
|
||||
if isSuccess {
|
||||
println( "老码请你吃火锅!")
|
||||
}
|
||||
```
|
||||
|
||||
####运行结果如下:
|
||||
|
||||
1. Hello, World!
|
||||
2. 老码请你吃火锅!
|
||||
3. Program ended with exit code: 0
|
||||
|
||||
```
|
||||
Hello, World!
|
||||
老码请你吃火锅!
|
||||
Program ended with exit code: 0
|
||||
```
|
||||
#####注意:
|
||||
- 如果小伙伴们现在用的是Beta版的Xcode,注意苹果官方Blog中,在代码第17行如果在Xcode Beta4下是错误的,这里的协议是,LogicValue而不是BooleanVue,所以记得看错误提示才是好习惯。
|
||||
- 注意代码第34行,完美支持if判断,且输出结果为“老码请你吃火锅”,老码也是说说而已,请不要当真。
|
||||
@ -165,34 +175,35 @@ OCBool现在只能用bool类型初始化,而不能直接返回bool型,小火
|
||||
小伙伴们,江湖风险,门派众多,老码有自己的OCBool类型,可能嵩山少林有自己的SSBool类型,甚至连郭美美都可能有自己的MMBool类型,所以OCBool必须能够识别这些类型,这些各门各派的类型,只要支持LogicValue协议,就应该可以被识别,看老码怎么做,
|
||||
|
||||
#####代码示例如下:
|
||||
```swift
|
||||
extension OCBool{
|
||||
init( _ v: LogicValue )
|
||||
{
|
||||
if v.getLogicValue(){
|
||||
self = .ocTrue
|
||||
}
|
||||
else{
|
||||
self = .ocFalse
|
||||
}
|
||||
}
|
||||
|
||||
1. extension OCBool{
|
||||
2. init( _ v: LogicValue )
|
||||
3. {
|
||||
4. if v.getLogicValue(){
|
||||
5. self = .ocTrue
|
||||
6. }
|
||||
7. else{
|
||||
8. self = .ocFalse
|
||||
9. }
|
||||
10. }
|
||||
11.
|
||||
12. }
|
||||
13.
|
||||
14. var mmResult: Bool = true
|
||||
15. var ocResult:OCBool = OCBool(mmResult)
|
||||
16.
|
||||
17.
|
||||
18. if ocResult {
|
||||
19. println( "老码没钱,郭美美请你吃火锅!")
|
||||
20. }
|
||||
}
|
||||
|
||||
var mmResult: Bool = true
|
||||
var ocResult:OCBool = OCBool(mmResult)
|
||||
|
||||
|
||||
if ocResult {
|
||||
println( "老码没钱,郭美美请你吃火锅!")
|
||||
}
|
||||
```
|
||||
|
||||
#####代码运行结果如下:
|
||||
|
||||
1. Hello, World!
|
||||
2. 老码没钱,郭美美请你吃火锅!
|
||||
3. Program ended with exit code: 0
|
||||
|
||||
```
|
||||
Hello, World!
|
||||
老码没钱,郭美美请你吃火锅!
|
||||
Program ended with exit code: 0
|
||||
```
|
||||
漂亮!我们的OCBool类型现在支持了所有的逻辑变量初始化。
|
||||
|
||||
#####注意:
|
||||
@ -202,71 +213,73 @@ OCBool现在只能用bool类型初始化,而不能直接返回bool型,小火
|
||||
####完善OCBool的布尔基因体系:
|
||||
小伙伴们,bool类型的价值就是在于各种判断,诸如==,!=, &,|,^,!,以及各种组合逻辑运算,我们OCBool也要具备这些功能,否则就会基因缺陷,且看老码如何实现:
|
||||
|
||||
1. extension OCBool: Equatable{
|
||||
2. }
|
||||
3.
|
||||
4. //支持等值判断运算符
|
||||
5. func ==( left: OCBool, right: OCBool )->Bool{
|
||||
6. switch (left, right){
|
||||
7. case (.ocTrue, .ocTrue):
|
||||
8. return true
|
||||
9. default:
|
||||
10. return false
|
||||
11. }
|
||||
12. }
|
||||
13. //支持位与运算符
|
||||
14. func &( left:OCBool, right: OCBool)->OCBool{
|
||||
15.
|
||||
16. if left{
|
||||
17. return right
|
||||
18. }
|
||||
19. else{
|
||||
20. return false
|
||||
21. }
|
||||
22. }
|
||||
23. //支持位或运算符
|
||||
24. func |( left:OCBool, right: OCBool)->OCBool{
|
||||
25.
|
||||
26. if left{
|
||||
27. return true
|
||||
28. }
|
||||
29. else{
|
||||
30. return right
|
||||
31. }
|
||||
32. }
|
||||
33.
|
||||
34. //支持位异或运算符
|
||||
35. func ^( left:OCBool, right: OCBool)->OCBool{
|
||||
36. return OCBool( left != right )
|
||||
37. }
|
||||
38. //支持求反运算符
|
||||
39. @prefix func !( a:OCBool )-> OCBool{
|
||||
40. return a ^ true
|
||||
41. }
|
||||
42. //支持组合求与运算符
|
||||
43. func &= (inout left:OCBool, right:OCBool ){
|
||||
44. left = left & right
|
||||
45. }
|
||||
46.
|
||||
47.
|
||||
48. var isHasMoney:OCBool = true
|
||||
49. var isHasWife:OCBool = true
|
||||
50. var isHasHealty:OCBool = true
|
||||
51. var isHasLover:OCBool = true
|
||||
52.
|
||||
53. isHasMoney != isHasHealty
|
||||
54. isHasHealty == isHasMoney
|
||||
55. isHasWife ^ isHasLover
|
||||
56. isHasWife = !isHasLover
|
||||
57.
|
||||
58. if (isHasMoney | isHasHealty) & isHasHealty{
|
||||
59. println( "人生赢家,就像老码一样!")
|
||||
60. }else
|
||||
61. {
|
||||
62. println("人生最苦的事事,人死了钱没花了,人生最苦的事是,人活着,钱没了!")
|
||||
63. }
|
||||
```swift
|
||||
extension OCBool: Equatable{
|
||||
}
|
||||
|
||||
好了,到这里就到这里了,窗外的雷声叫醒了老码,现在应该去吃饭了,以上老码给大家展示了如果制造一个自己的类型,记得老码的示例是在Xcode6 Beta4下测试的,至于Beta5的改变还没有涉及,小伙伴们要好生练习,以后各种自定类型都是基于这个思想。还有这个章节不是老码的原创,老码认真的阅读了苹果的官方博客,且自己的练习总结,如果小伙伴们费了吃奶的劲还是看不懂,请找度娘谷歌,还是看不懂请到老码官方微博:http://weibo.com/u/5241713117咆哮。
|
||||
//支持等值判断运算符
|
||||
func ==( left: OCBool, right: OCBool )->Bool{
|
||||
switch (left, right){
|
||||
case (.ocTrue, .ocTrue):
|
||||
return true
|
||||
default:
|
||||
return false
|
||||
}
|
||||
}
|
||||
//支持位与运算符
|
||||
func &( left:OCBool, right: OCBool)->OCBool{
|
||||
|
||||
if left{
|
||||
return right
|
||||
}
|
||||
else{
|
||||
return false
|
||||
}
|
||||
}
|
||||
//支持位或运算符
|
||||
func |( left:OCBool, right: OCBool)->OCBool{
|
||||
|
||||
if left{
|
||||
return true
|
||||
}
|
||||
else{
|
||||
return right
|
||||
}
|
||||
}
|
||||
|
||||
//支持位异或运算符
|
||||
func ^( left:OCBool, right: OCBool)->OCBool{
|
||||
return OCBool( left != right )
|
||||
}
|
||||
//支持求反运算符
|
||||
@prefix func !( a:OCBool )-> OCBool{
|
||||
return a ^ true
|
||||
}
|
||||
//支持组合求与运算符
|
||||
func &= (inout left:OCBool, right:OCBool ){
|
||||
left = left & right
|
||||
}
|
||||
|
||||
|
||||
var isHasMoney:OCBool = true
|
||||
var isHasWife:OCBool = true
|
||||
var isHasHealty:OCBool = true
|
||||
var isHasLover:OCBool = true
|
||||
|
||||
isHasMoney != isHasHealty
|
||||
isHasHealty == isHasMoney
|
||||
isHasWife ^ isHasLover
|
||||
isHasWife = !isHasLover
|
||||
|
||||
if (isHasMoney | isHasHealty) & isHasHealty{
|
||||
println( "人生赢家,就像老码一样!")
|
||||
}else
|
||||
{
|
||||
println("人生最苦的事事,人死了钱没花了,人生最苦的事是,人活着,钱没了!")
|
||||
}
|
||||
```
|
||||
|
||||
好了,到这里就到这里了,窗外的雷声叫醒了老码,现在应该去吃饭了,以上老码给大家展示了如果制造一个自己的类型,记得老码的示例是在Xcode6 Beta4下测试的,至于Beta5的改变还没有涉及,小伙伴们要好生练习,以后各种自定类型都是基于这个思想。还有这个章节不是老码的原创,老码认真的阅读了苹果的官方博客,且自己的练习总结,如果小伙伴们费了吃奶的劲还是看不懂,请找度娘谷歌,还是看不懂请到老码官方微博:http://weibo.com/u/5241713117咆哮。
|
||||
|
||||
|
||||
|
||||
|
||||
@ -15,25 +15,27 @@ Swift里面的类型分为两种:
|
||||
|
||||
值类型和引用类型最基本的分别在复制之后的结果。当一个值类型被复制的时候,相当于创造了一个完全独立的实例,这个实例保有属于自己的独有数据,数据不会受到其他实例的数据变化影响:
|
||||
|
||||
```swift
|
||||
// 下面是一个值类型的例子
|
||||
struct S { var data: Int = -1 }
|
||||
var a = S()
|
||||
var b = a // b是a的拷贝
|
||||
a.data = 42 // 更改a的数据,b的不受影响
|
||||
println("\(a.data), \(b.data)") // 输出结果 "42, -1"
|
||||
```
|
||||
|
||||
值类型就好像身份证复印件一样,复印出来之后,修改原件上面的内容,复印件上的内容不会变。
|
||||
|
||||
另一方面,复制一个引用类型的时候,实际上是默默地创造了一个共享的实例分身,两者是共用一套数据。因此修改其中任何一个实例的数据,也会影响到另外那个。
|
||||
|
||||
|
||||
```swift
|
||||
// 下面是一个引用类型的例子
|
||||
class C { var data: Int = -1 }
|
||||
var x = C()
|
||||
var y = x // y是x的拷贝
|
||||
x.data = 42 // 更改x的数据,等于同时修改了y
|
||||
println("\(x.data), \(y.data)") // 输出结果 "42, 42"
|
||||
|
||||
```
|
||||
|
||||
|
||||
### Mutation(变异)在安全中扮演的角色
|
||||
|
||||
Reference in New Issue
Block a user