Swift類型創(chuàng)建之自定義一個(gè)類型詳解
小伙伴們,Swift中的Bool類型有著非常重要的語法功能,并支撐起了整個(gè)Swift體系中的邏輯判斷體系,經(jīng)過老碼的研究和學(xué)習(xí), Bool類型本身其實(shí)是對(duì)基礎(chǔ)Boolean類型封裝,小伙伴們可能咬著手指頭問老碼,怎么一會(huì)Bool類型,一會(huì)Boolean類型,其區(qū)別在于,前者是基于枚舉的組合類型,而后者則是基本類型,只有兩種true和false。
####自定義原型
接下老碼根據(jù)Bool的思想來創(chuàng)建一個(gè)OCBool類型,來讓小伙伴們了解一下Swift中到底是怎么玩兒的。
來我們先看一下OCBool的定義。
#####代碼示例如下:
enum OCBool{
case ocTrue
case ocFalse
}
#####注意:
代碼中第2行和第3行,可以合并到一行寫,如蘋果官方Blog所寫的一樣
代碼中命名需要注意:OCBool是類型名,所以首字母必須大寫,而case中的ocTrue和ocFalse是小類型則需要首字母小寫。
####實(shí)現(xiàn)默認(rèn)值
行,我們給了一個(gè)漂亮的定義,不過按照傳統(tǒng)語言的經(jīng)驗(yàn),Bool值默認(rèn)情況下是假, 所以我們的OCBool也應(yīng)該如此,我們使用類型擴(kuò)展技術(shù)增加這個(gè)默認(rèn)特性:
extension OCBool{
init(){
self =.ocFalse
}
}
#####注意:
●代碼中第1行:extension關(guān)鍵字,非常強(qiáng)大,小伙伴們可以通過此創(chuàng)造出許多好玩的東西,建議各位去Github上看一個(gè)名為“Swiftz”的項(xiàng)目,它將擴(kuò)展用到了極致。
●代碼中第3行:self = .ocFalse語法,剛?cè)腴T的小伙伴們很迷糊,為什么會(huì)有奇怪的點(diǎn)語法,因?yàn)榇笈hris在Swift中增加了類型智能推斷功能,在蘋果Blog中,提到了“Context”概念,就是這個(gè)意思,因?yàn)檫@行語句是在枚舉OCBool中的,其上下文就是OCBool的定義體,編譯器當(dāng)然知道.ocFalse就是OCBool.ocFalse了,所以這里直接點(diǎn)語法,非常整齊。
現(xiàn)在我們可以使用如下方法使用這個(gè)Bool類型。
#####代碼示例如下:
var result:OCBool = OCBool()
var result1:OCBool = .ocTrue
####支持基本布爾型初始化
正如上述代碼所述,我們只能通過類型或者枚舉項(xiàng)目賦值,這是組合類型的用法,但是編碼的日子里,我們總是希望和true,false直接打交道,也就是說,我們希望這么做,
代碼示例如下:
var isSuccess:OCBool = true
如果小伙伴們直接這么用,則會(huì)出現(xiàn)如下錯(cuò)誤:
/Users/tyrion-OldCoder/Documents/Learning/BoolType/BoolType/main.swift:30:24: Type 'OCBool' does not conform to protocol 'BooleanLiteralConvertible'
編譯器咆哮的原因是,我們的類型沒有遵從“布爾字面量轉(zhuǎn)換協(xié)議”,接下來修正這個(gè)問題,
#####代碼示例如下:
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行是重點(diǎn),我的類型OCBool支持了BooleanLiteralConvertible協(xié)議,這個(gè)協(xié)到底是干什么的呢,小伙伴們?cè)赬code代碼編輯器,按住Command鍵,然后點(diǎn)擊第11行中的BooleanLiteralConvertible協(xié)議名,則會(huì)進(jìn)入它的定義,
#####其定義如下:
protocol BooleanLiteralConvertible {
typealias BooleanLiteralType
class func convertFromBooleanLiteral(value: BooleanLiteralType) -> Self
}
這個(gè)定義中有個(gè)類方法convertFromBooleanLiteral,它的參數(shù)為BooleanLiteralType類型,也就是我傳入的Bool類型, 且返回值為實(shí)現(xiàn)這個(gè)協(xié)議的類型本身,在我們的OCBool類型中,其返回值就是OCBool本身。經(jīng)過這個(gè)定義,我們可以直接對(duì)OCBool類型直接進(jìn)行布爾字面量初始化了。
####支持Bool類型判斷
小伙伴們不安分, 肯定想著我怎么用它實(shí)現(xiàn)邏輯判斷,所以如果你這么寫,
#####代碼示例如下:
var isSuccess:OCBool = true
if isSuccess {
println( "老碼請(qǐng)你吃火鍋!")
}
你永遠(yuǎn)吃不到老碼的火鍋,因?yàn)檫@里編譯器會(huì)咆哮:
/Users/tyrion-OldCoder/Documents/Learning/BoolType/BoolType/main.swift:27:4: Type 'OCBool' does not conform to protocol 'LogicValue'
OCBool現(xiàn)在只能用bool類型初始化,而不能直接返回bool型,小火把們還記得在《老碼說編程之白話Swift江湖》中,老碼多次提到,媽媽再也不擔(dān)心我們 if a = 1{}的寫法了, 因?yàn)榈忍?hào)不支持值返回了, 所以在if判斷是后面的條件必須有返回值,OCBool沒有,所以編譯器哭了。我們解決這個(gè)問題。
#####代碼示例如下:
import Foundation
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( "老碼請(qǐng)你吃火鍋!")
}
####運(yùn)行結(jié)果如下:
Hello, World!
老碼請(qǐng)你吃火鍋!
Program ended with exit code: 0
#####注意:
●如果小伙伴們現(xiàn)在用的是Beta版的Xcode,注意蘋果官方Blog中,在代碼第17行如果在Xcode Beta4下是錯(cuò)誤的,這里的協(xié)議是,LogicValue而不是BooleanVue,所以記得看錯(cuò)誤提示才是好習(xí)慣。
●注意代碼第34行,完美支持if判斷,且輸出結(jié)果為“老碼請(qǐng)你吃火鍋”,老碼也是說說而已,請(qǐng)不要當(dāng)真。
####支持兼容各們各派的類型
小伙伴們,江湖風(fēng)險(xiǎn),門派眾多,老碼有自己的OCBool類型,可能嵩山少林有自己的SSBool類型,甚至連郭美美都可能有自己的MMBool類型,所以O(shè)CBool必須能夠識(shí)別這些類型,這些各門各派的類型,只要支持LogicValue協(xié)議,就應(yīng)該可以被識(shí)別,看老碼怎么做,
#####代碼示例如下:
extension OCBool{
init( _ v: LogicValue )
{
if v.getLogicValue(){
self = .ocTrue
}
else{
self = .ocFalse
}
}
}
var mmResult: Bool = true
var ocResult:OCBool = OCBool(mmResult)
if ocResult {
println( "老碼沒錢,郭美美請(qǐng)你吃火鍋!")
}
#####代碼運(yùn)行結(jié)果如下:
Hello, World!
老碼沒錢,郭美美請(qǐng)你吃火鍋!
Program ended with exit code: 0
漂亮!我們的OCBool類型現(xiàn)在支持了所有的邏輯變量初始化。
#####注意:
●代碼中第2行:“_”下橫杠的用法,這是一個(gè)功能強(qiáng)大的小強(qiáng),在此的目的是屏蔽外部參數(shù)名,所以小伙伴們可以直接:var ocResult:OCBool = OCBool(mmResult)而不是:var ocResult:OCBool = OCBool(v: mmResult),小伙伴們驚呆了!這個(gè)init函數(shù)中本來就沒有外部參數(shù)名啊,還記得老碼在書里說過沒,Swift的初始化函數(shù)會(huì)默認(rèn)使用內(nèi)部參數(shù)名,作為外部參數(shù)名。
####完善OCBool的布爾基因體系:
小伙伴們,bool類型的價(jià)值就是在于各種判斷,諸如==,!=, &,|,^,!,以及各種組合邏輯運(yùn)算,我們OCBool也要具備這些功能,否則就會(huì)基因缺陷,且看老碼如何實(shí)現(xiàn):
extension OCBool: Equatable{
}
//支持等值判斷運(yùn)算符
func ==( left: OCBool, right: OCBool )->Bool{
switch (left, right){
case (.ocTrue, .ocTrue):
return true
default:
return false
}
}
//支持位與運(yùn)算符
func &( left:OCBool, right: OCBool)->OCBool{
if left{
return right
}
else{
return false
}
}
//支持位或運(yùn)算符
func |( left:OCBool, right: OCBool)->OCBool{
if left{
return true
}
else{
return right
}
}
//支持位異或運(yùn)算符
func ^( left:OCBool, right: OCBool)->OCBool{
return OCBool( left != right )
}
//支持求反運(yùn)算符
@prefix func !( a:OCBool )-> OCBool{
return a ^ true
}
//支持組合求與運(yùn)算符
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("人生最苦的事事,人死了錢沒花了,人生最苦的事是,人活著,錢沒了!")
}
好了,到這里就到這里了,窗外的雷聲叫醒了老碼,現(xiàn)在應(yīng)該去吃飯了,以上老碼給大家展示了如果制造一個(gè)自己的類型,記得老碼的示例是在Xcode6 Beta4下測(cè)試的,至于Beta5的改變還沒有涉及,小伙伴們要好生練習(xí),以后各種自定類型都是基于這個(gè)思想。還有這個(gè)章節(jié)不是老碼的原創(chuàng),老碼認(rèn)真的閱讀了蘋果的官方博客,且自己的練習(xí)總結(jié)
相關(guān)文章
詳解swift中xcworkspace多項(xiàng)目管理
給大家詳細(xì)講解了IOS開發(fā)中swift語言xcworkspace多項(xiàng)目管理的方法和介紹,一起參考一下。2017-11-11swift 錯(cuò)誤處理do catch try try!使用詳解
這篇文章主要介紹了swift 錯(cuò)誤處理do catch try try!使用詳解的相關(guān)資料,需要的朋友可以參考下2023-03-03swift實(shí)現(xiàn)簡(jiǎn)易計(jì)算器項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了swift實(shí)現(xiàn)簡(jiǎn)易計(jì)算器項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01switch實(shí)現(xiàn)一個(gè)兩數(shù)的運(yùn)算代碼示例
這篇文章主要介紹了switch實(shí)現(xiàn)一個(gè)兩數(shù)的運(yùn)算代碼示例,需要的朋友可以參考下2017-06-06