Swift 2.0初探:值得注意的新特性

轉眼間,Swift已經一歲多了,這門新鮮、語法時尚、類型安全、執行速度更快的語言已經漸漸的深入廣大開發者的心。我同樣也是非常喜愛這門新的編程語言。

今年6月,一年一度的WWDC大會如期而至,在大會上Apple發布了Swift 2.0,引入了很多新的特性,以幫助開發者能更快,更簡單的構建應用。我在這里也說道說道Swift 2.0中值得大家注意的新特性。

guard語句

guard語句和if語句有點類似,都是根據其關鍵字之后的表達式的布爾值決定下一步執行什么。但與if語句不同的是,guard語句只會有一個代碼塊,不像if語句可以if else多個代碼塊。

那么guard語句的作用到底是什么呢?顧名思義,就是守護。guard語句判斷其后的表達式布爾值為false時,才會執行之后代碼塊里的代碼,如果為true,則跳過整個guard語句,我們舉例來看看。

我們以今年高考為例,在進入考場時一般都會檢查身份證和準考證,我們寫這樣一個方法:

func?checkup(person:?[String:?String!])?{//?檢查身份證,如果身份證沒帶,則不能進入考場guard?let?id?=?person["id"]?else?{print("沒有身份證,不能進入考場!")return}//?檢查準考證,如果準考證沒帶,則不能進入考場guard?let?examNumber?=?person["examNumber"]?else?{print("沒有準考證,不能進入考場!")return}//?身份證和準考證齊全,方可進入考場print("您的身份證號為:\(id),準考證號為:\(examNumber)。請進入考場!")}
checkup(["id":?"123456"])?//?沒有準考證,不能進入考場!
checkup(["examNumber":?"654321"])?//?沒有身份證,不能進入考場!
checkup(["id":?"123456",?"examNumber":?"654321"])?//?您的身份證號為:123456,準考證號為:654321。請進入考場!

上述代碼中的第一個guard語句用于檢查身份證,如果檢查到身份證沒帶,也就是表達式為false時,執行大括號里的代碼,并返回。第二個guard語句則檢查準考證。

如果兩證齊全,則執行最后一個打印語句,上面的兩個guard語句大括號內的代碼都不會執行,因為他們表達式的布爾值都是true。

這里值得注意的是,id和examNumber可以在guard語句之外使用,也就是說當guard對其表達式進行驗證后,id和examNumber可在整個方法的作用域中使用,并且是解包后的。

我們再用if else語句寫一個類似的方法:

func?checkupUseIf(person:?[String:?String!])?{if?let?id?=?person["id"],?let?examNumber?=?person["examNumber"]?{print("您的身份證號為:\(id),準考證號為:\(examNumber)。請進入考場!")}?else?{print("證件不齊全,不能進入考場!")}print("您的身份證號為:\(id),準考證號為:\(examNumber)")??//?報異常}
checkupUseIf(["id":?"123456"])?//?證件不齊全,不能進入考場!
checkupUseIf(["examNumber":?"654321"])?//?證件不齊全,不能進入考場!
checkupUseIf(["id":?"123456",?"examNumber":?"654321"])?//?您的身份證號為:123456,準考證號為:654321。請進入考場!

我們可以看到用if else實現的方法顯然不如guard實現的那么精準。而且id和examNumber的作用域只限在if的第一個大括號內,超出這個作用域編譯就會報錯。

通過上述兩個小例子不難看出,guard語句正如一個稱職的守衛,層層把關,嚴防一切不允許發生的事,并且讓代碼具有更高的可讀性,非常棒。

異常處理

在Swift 1.0時代是沒有異常處理和拋出機制的,如果要處理異常,要么使用if else語句或switch語句判斷處理,要么使用閉包形式的回調函數處理,再要么就使用NSError處理。以上這些方法都不能像Java中的try catch異常控制語句那樣行如流水、從容不迫的處理異常,而且也會降低代碼的可讀性。當Swift 2.0到來后,一切都不一樣了。

在Swift 2.0中Apple提供了使用throws、throw、try、do、catch這五個關鍵字組成的異常控制處理機制。下面我們來舉例看看如何使用,我用使用手機刷朋友圈為例。

首先我們需要定義異常枚舉,在Swift 2.0中Apple提供了ErrorType協議需要我們自定義的異常枚舉遵循:

enum?WechatError:?ErrorType?{case?NoBattery?//?手機沒電case?NoNetwork?//?手機沒網case?NoDataStream?//?手機沒有流量
}

我們定義了導致不能刷微信的錯誤枚舉’wechatError。然后定義一個檢查是否可以刷微信的方法checkIsWechatOk():

func?checkIsWechatOk(isPhoneHasBattery:?Bool,?isPhoneHasNetwork:?Bool,?dataStream:?Int)?throws?{guard?isPhoneHasBattery?else?{throw?WechatError.NoBattery}guard?isPhoneHasNetwork?else?{throw?WechatError.NoNetwork}guard?dataStream?>?50?else?{throw?WechatError.NoDataStream}}

這里注意,在方法名后有throws關鍵字,意思為該方法產生的異常向上層拋出。在方法體內使用guard語句對各種狀態進行判斷,然后使用throw關鍵字拋出對應的異常。然后我們定義刷微信的方法:

func?playWechat(isPhoneHasBattery:?Bool,?isPhoneHasNetwork:?Bool,?dataStream:?Int)?{do?{try?checkIsWechatOk(isPhoneHasBattery,?isPhoneHasNetwork:?isPhoneHasNetwork,?dataStream:?dataStream)print("放心刷,刷到天昏地暗!")}?catch?WechatError.NoBattery?{print("手機都沒電,刷個鬼啊!")}?catch?WechatError.NoNetwork?{print("沒有網絡哎,洗洗玩單機吧!")}?catch?WechatError.NoDataStream?{print("沒有流量了,去蹭Wifi吧!")}?catch?{print("見鬼了!")}}
playWechat(true,?isPhoneHasNetwork:?true,?dataStream:?60)?//?放心刷,刷到天昏地暗!
playWechat(true,?isPhoneHasNetwork:?false,?dataStream:?60)?//?沒有網絡哎,洗洗玩單機吧!
playWechat(false,?isPhoneHasNetwork:?true,?dataStream:?60)?//?手機都沒電,刷個鬼啊!
playWechat(true,?isPhoneHasNetwork:?true,?dataStream:?30)?//?沒有流量了,去蹭Wifi吧!

上述的代碼示例中,首先檢查是否可以刷微信的方法前使用try關鍵字,表示允許該方法拋出異常,然后使用了do catch控制語句捕獲拋出的異常,進而做相關的邏輯處理。

這套異常處理機制使Swift更加的全面和安全,并且提高了代碼的可讀性,非常棒。

協議擴展

在Swift 1.0 時代,協議(Protocol)基本上類似一個接口,定義若干屬性和方法,供類、結構體、枚舉遵循和實現。在Swift 2.0中,可以對協議進行屬性或者方法的擴展,和擴展類與結構體類似。這讓我們開啟了面向協議編程的篇章。

Swift 中,大多數基礎對象都遵循了CustomStringConvertible協議,比如Array、Dictionary(Swift 1.0中的Printable協議),該協議定義了description方法,用于print方法打印對象。現在我們對該協議擴展一個方法,讓其打印出 大寫的內容:

var?arr?=?["hello",?"world"]
print(arr.description)?//?"[hello,?world]"
extension?CustomStringConvertible?{var?upperDescription:?String?{return?"\(self.description.uppercaseString)"}
}
print(arr.upperDescription)?//?"[HELLO,?WORLD]"

如果在Swfit 1.0時代,要想達到上述示例的效果,那么我們需要分別對Array、Dictionary進行擴展,所以協議的擴展極大的提高了我們的編程效率,也同樣使代碼更簡潔和易讀。

打印語句的改變

在Swift1中,有'println()'和'print()'兩個在控制臺打印語句的方法,前者是換行打印,后者是連行打印。在Swift2中,'println()'已成為過去,取而代之的是他倆的結合體。如果你想做換行打印,現在需要這樣寫:

print("我要換行!",?appendNewline:?true)

available檢查

作為iOS開發者,誰都希望使用最新版本iOS的Api進行開發,省事省力。但常常事與愿違,因為我們經常需要適配老版本的iOS,這就會面臨一個問題,一些新特性特性或一些類無法在老版本的iOS中使用,所以在編碼過程中經常會對iOS的版本做以判斷,就像這樣:

if?NSClassFromString("NSURLQueryItem")?!=?nil?{//?iOS?8或更高版本
}?else{//?iOS8之前的版本
}

以上這只是一種方式,在Swift 2.0之前也沒有一個標準的模式或機制幫助開發者判斷iOS版本,而且容易出現疏漏。在Swift 2.0到來后,我們有了標準的方式來做這個工作:

if?#available(iOS?8,?*)?{//?iOS?8或更高版本let?queryItem?=?NSURLQueryItem()}?else?{//?iOS8之前的版本}

這個特性讓我們太幸福。

do-while語句重命名

經典的do-while語句改名了,改為了repeat-while:

var?i?=?0
repeat?{i++print(i)
}?while?i?<?10

個人感覺更加直觀了。

defer關鍵字

在一些語言中,有try/finally這樣的控制語句,比如Java。這種語句可以讓我們在finally代碼塊中執行必須要執行的代碼,不管之前怎樣的興風作浪。在Swift 2.0中,Apple提供了defer關鍵字,讓我們可以實現同樣的效果。

func?checkSomething()?{print("CheckPoint?1")doSomething()print("CheckPoint?4")}
func?doSomething()?{print("CheckPoint?2")defer?{print("Clean?up?here")}print("CheckPoint?3")}
checkSomething()?//?CheckPoint?1,?CheckPoint?2,?CheckPoint?3,?Clean?up?here,?CheckPoint?4

上述示例可以看到,在打印出“CheckPoint 2”之后并沒有打印出“Clean up here”,而是“CheckPoint 3”,這就是defer的作用,它對進行了print("Clean up here")延遲。我們再來看一個I/O的示例:

//?偽代碼
func?writeSomething()?{let?file?=?OpenFile()let?ioStatus?=?fetchIOStatus()guard?ioStatus?!=?"error"?else?{return}file.write()closeFile(file)}

上述示例是一個I/O操作的偽代碼,如果獲取到的ioStatus正常,那么該方法沒有問題,如果ioStatus取到的是 error,那么會被guard語句抓到執行return操作,這樣的話closeFile(file)就永遠都不會執行了,一個嚴重的Bug就這樣產生 了。下面我們看看如何用defer來解決這個問題:

//?偽代碼
func?writeSomething()?{let?file?=?OpenFile()defer?{closeFile(file)}let?ioStatus?=?fetchIOStatus()guard?ioStatus?!=?"error"?else?{return}file.write()}

我們將closeFile(file)放在defer代碼塊里,這樣即使ioStatus為error,在執行return前會先執行defer里的代碼,這樣就保證了不管發生什么,最后都會將文件關閉。

defer又一個保證我們代碼健壯性的特性,我非常喜歡。

Swift 2.0中的新特性當然不止以上這些,但窺一斑可見全豹,Swift 2.0努力將更快、更安全做到極致,這是開發人員的福音,讓我們盡情享受這門美妙的語言吧。

轉載于:https://www.cnblogs.com/Free-Thinker/p/5151089.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/256020.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/256020.shtml
英文地址,請注明出處:http://en.pswp.cn/news/256020.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Android 自定義WebView彈窗及屏蔽彈窗

額&#xff0c;還是那個WebView的問題&#xff0c;內核已換成騰訊X5內核&#xff0c;所以接下來的內容會有一些X5內核的方法。但我們的H5是不能改的&#xff0c;還是只有委屈我們自己。先看看H5自帶的彈窗 這樣子的彈窗在不同的手機上呈現的可能是不同的效果&#xff0c;效果不…

【圖像處理】——Python實現two_pass方法來進行連通域的提取

目錄 一、相關知識 1、two_pass算法思想 2、并查集算法 二、自定義的two_pass算法

C++ 多線程使用future傳遞異常

如果 std::async 調用的函數拋出異常&#xff0c;那么這個異常會被存儲在值的位置&#xff0c;同時 future 變為 ready ,如果調用 get() 會重新拋出存儲的異常。 Note: 標準并沒有指定原來的異常對象是被重新拋出或者拷貝后拋出&#xff0c;不同的編譯器會做不同的選擇。 對于 …

期貨黃金與現貨黃金比較

現貨黃金與期貨黃金是目前市場上最熱門的黃金投資方式&#xff0c;與國內任何的金融投資品相比&#xff0c;都具有一定的優勢。 其實金投網小編覺得現貨黃金與期貨黃金最主要的不同點是這個&#xff1a;期貨黃金做的是國內市場&#xff0c;同股票市場一樣&#xff0c;里面有莊家…

DNS域傳送漏洞

0x00 相關背景介紹 Dns是整個互聯網公司業務的基礎&#xff0c;目前越來越多的互聯網公司開始自己搭建DNS服務器做解析服務&#xff0c;同時由于DNS服務是基礎性服務非常重要&#xff0c;因此很多公司會對DNS服務器進行主備配置而DNS主備之間的數據同步就會用到dns域傳送&#…

封裝之--通過類中公有方法訪問私有成員變量

如何在ClassB中訪問ClassA的私有成員變量&#xff1f;&#xff08;典型的封裝案例&#xff09; 通過在ClassA中定義公有的成員方法&#xff0c;然后&#xff0c;在ClassB中通過ClassA的對象調用ClassA中的公有方法&#xff0c;來訪問ClassA中的私有成員變量。 轉載于:https://w…

匹配物鏡放大倍數與相機像元尺寸

通常來說&#xff0c;相機內部的CCD或者CMOS傳感器上都有感光陣列&#xff0c;由一個一個的感光元件構成&#xff0c;每一個感光元件負責完成光電轉換的過程。簡單理解&#xff0c;一個感光元件可以認為就是一個像素(pixel)或像元(pel)。像元具有一定尺寸&#xff0c;如果像的尺…

2016/11/10 kettle概述

ETL(Extract-Transform-Load&#xff0c;即抽取&#xff0c;轉換&#xff0c;加載)&#xff0c;數據倉庫技術&#xff0c;是用來處理將數據從來源&#xff08;以前做的項目&#xff09;經過抽取&#xff0c;轉換&#xff0c;加載到達目的端&#xff08;正在做的項目&#xff09…

【深度學習】——非極大值抑制(nms/soft-nms)

目錄 一、相關概念 1、iou 1&#xff09;理論計算 2&#xff09;Python代碼&#xff08;代碼參考yolov3模型util.py文件&#xff09; 2、nms 1)基本思路 2&#xff09;標準nms和soft-nms 3&#xff09;Python代碼實現&#xff08;yolov3中util.py文件&#xff0c;增加了…

移動服務安全現狀分析!

2019獨角獸企業重金招聘Python工程師標準>>> 由于Android開源的環境&#xff0c;導致Android的整體環境都存在很多不安全的因素&#xff0c;同時用戶在移動APP客戶端的便捷應用&#xff0c;也給用戶帶來了巨大的安全隱患。未經過移動服務安全加固的APP存在被靜態反編…

封裝不同類模板的隨機數生成器

最近準備刷題&#xff0c;打算簡單封裝下隨機數生成器&#xff0c;方便產生測試數據。C11的STL提供了很多分布類型&#xff0c;我比較常用的是均勻分布&#xff0c;均勻分布的值有兩種類型&#xff0c;一類是整數&#xff0c;另一類是浮點數&#xff0c;STL根據值的類型定義了兩…

Log4j詳細設置說明

1. 動態的改變記錄級別和策略&#xff0c;即修改log4j.properties,不需要重啟Web應用&#xff0c;這需要在web.xml中設置一下。2. 把log文件定在 /WEB-INF/logs/ 而不需要寫絕對路徑。3. 可以把log4j.properties和其他properties一起放在/WEB-INF/ &#xff0c;而不是Class-Pat…

【機器學習】——卷積神經網絡(Keras)修改學習率(定值+自適應)

目錄 1、直接獲取現有模型的學習率 2、打印顯示學習率 3、調整學習率 1&#xff09;1.LearningRateScheduler 2&#xff09;ReduceLROnPlateau 利用tensorflow的Keras模塊我們可以建立我們自己定義的卷積神經網絡模型&#xff0c;但是一般不會觸碰到學習率這個問題&#x…

使用Docker構建你的第一個服務

1. 感受一下Docker的便捷 項目源碼 https://github.com/MoonShining/dockernize-grape-helloworld clone下來以后運行,cd到項目目錄下&#xff0c;運行 docker build -t api-sample . docker run -p 8080:8080 api-sample:latest 復制代碼就可以訪問localhost:8080看到效果了. …

MyEclipse連接MySQL

在官網http://www.mysql.com/downloads/下載數據庫連接驅動 本文中使用驅動版本為mysql-connector-java-5.1.40 一、創建一個java測試項目MySQLConnectorsTest 在項目下穿件一個lib文件夾用來存放MySQL驅動包。 右鍵驅動包build path進行add添加操作&#xff0c;打開Referenced…

在Windows系統中配置Google AddressSanitizer

Google AddressSanitizer簡介 AddressSanitizer (ASan) 是 C 和 C 的內存錯誤檢測軟件&#xff0c;它可以檢測&#xff1a; 釋放指針后繼續使用堆緩沖區溢出棧緩沖區溢出全局緩沖區溢出返回后繼續使用在范圍之外繼續使用初始化順序的bug內存泄漏 在 Windows 系統中&#xff…

【劍指offer】——求出一個正整數的質數因子(Python)

目錄 一、題目描述 二、思路 1、短除法 2、平方根法 一、題目描述 功能:輸入一個正整數&#xff0c;按照從小到大的順序輸出它的所有質因子&#xff08;重復的也要列舉&#xff09;&#xff08;如180的質因子為2 2 3 3 5 &#xff09; 最后一個數后面也要有空格 輸入描述…

Python字符串筆錄

python字符串操作實方法&#xff0c;包括了幾乎所有常用的python字符串操作&#xff0c;如字符串的替換、刪除、截取、復制、連接、比較、查找、分割等 1、去空格及特殊符號 >>> s 123 >>> s.strip() # 去除所有空格123 >>> s1 123 >>&…

第九周進度條

第六周學習進度條 星期一 星期二 星期三 星期四 星期五 星期六 星期日 所花時間 &#xff08;包括上課&#xff09; 無 上午8:00-9:50 無 無 下午4:00-5:00 下午3&#xff1a;00-5&#xff1a;30 代碼量&#xff08;行&#xff09; 0行 260行 0行 0行 0行 0…