雖然Swift出生才一年,但是它已經成為最流行的編程語言之一了。它的語法很簡單,以至于當它發布的時候,java script開發者感覺就像下圖一樣。
事實上,Swift是一種復雜的語言。它包含面向對象和函數方法這兩個方面,并且隨著新版本的發布在一直進化。
你可以用這些問題來測試應聘者關于Swift方面的知識水平,或者測試一下你自己。如果你不知道答案,沒關系,沒一個問題下面都有答案供你學習。
這些問題包含兩個方面:
筆試問題:通過電子郵件做一個編程測試是極好的,因為這涉及到寫大量的代碼,從代碼質量上可以看出一個人的水平。
面試問題:電話面試或者面對面面試也是很好的,因為對面試者來說口頭交流會更方面。
每個方面有分成三個等級:
初級:適合讀了一到兩本有關Swift的書,并且已經開始用Swift開發應用程序的初學者。
中級:適合那些對Swift語言的概念有深刻理解和強烈興趣的,并且一直在閱讀大量有關Swift的博客文章并進行實踐的中級工程師。
高級:適合那些以探索Swift語言知識為樂趣,挑戰自己,使用前言技術的人們。
假如你想回答這些問題,我建議你在回答這些問題之前,打開Playground運行一下這些問題的代碼。這些問題的答案都在Xcode 7.0 Beta 6 版本中測試過。
準備好了嗎?系好安全帶,現在就開始!
筆試問題
初學者
問題1、(Swift 1.0及其之后的版本的問題)有什么更好的方法來寫下面的for循環?
for var i = 0; i < 5; i++ {
print("Hello!")
}
答案:
for _ in 0...4 {
print("Hello!")
}
Swift 實現了兩個數組運算符closed operator 和 half-operator.前者包含數組中得所有值。例如:下面的例子包含從0到4得所有整數
0...4
half-operator不包含數組中的最后一個元素,下面的例子會得到的結果和上面的一樣:
0..<5
問題2– Swift 1.0 or later
思考下面的問題:
struct Tutorial {
var difficulty: Int = 1
}
var tutorial1 = Tutorial()
var tutorial2 = tutorial1
tutorial2.difficulty = 2
tutorial1.difficulty 和 tutorial2.difficulty的值分別是多少?假如Tutorial是一個類,會有什么不同?并說明原因。
答案:tutorial1.difficulty 的值是1,然而tutorial2.difficulty的值是2.
在Swift中結構體是值類型,他們的值是復制的而不是引用的。下面的一行代碼意思是復制了tutorial1的值并把它賦值給tutorial2:
var tutorial2 = tutorial1
從這一行開始,tutorial2值得改變并不影響tutorial1的值。
假如Tutorial是一個類,tutorial1.difficulty和tutorial2.difficulty的值將都會是2.在Swift中類對象都是引用類型。tutorial1屬性的任何改變將會反應到tutorial2上,反之亦然。
問題3 – Swift 1.0 or later
view1聲明成var類型,view2聲明let類型。這里有什么區別嗎?下面的最后一行代碼能編譯嗎?
import UIKit
var view1 = UIView()
view1.alpha = 0.5
let view2 = UIView()
view2.alpha = 0.5 // Will this line compile?
答案:view1是個變量可以重新賦值給一個新的實例化的UIView對象。使用let你只賦值一次,所以下面的代碼是不能編譯的:
view2 = view1 // Error: view2 is immutable
但是UIView是一個引用類型的類,所以你可以改變view2的屬性,也就是說最后一行代碼是可以編譯的:
let view2 = UIView()
view2.alpha = 0.5 // Yes!
問題4 – Swift 1.0 or later
下面的代碼是把數組里面的名字按字母的順序排序,看上去比較復雜。盡最大的可能簡化閉包里的代碼。
let animals = ["fish", "cat", "chicken", "dog"]
let sortedAnimals = animals.sort { (one: String, two: String) -> Bool in
return one < two
}
答案:
第一個簡化的是參數。系統的參數類型推斷功能,可以計算出閉包里面參數的類型,所以你不必定義參數的類型:
let sortedAnimals = animals.sort { (one, two) -> Bool in return one < two }
函數返回值也可以被推斷出來,所以簡化掉,代碼變為:
let sortedAnimals = animals.sort { (one, two) in return one < two }
這個$i 符號可以代替參數名字,代碼進一步簡化為:
let sortedAnimals = animals.sort { return $0 < $1 }
在一個獨立的閉包內,return這個關鍵字是可以省略的。最后聲明的返回值就是閉包的返回值:
let sortedAnimals = animals.sort { $0 < $1 }
這簡化很多了,但是我們不能止步于此!
對于字符串,有一個定義如下的比較函數:
func Bool
這個簡單的小函數可以使你的代碼簡潔如下:
let sortedAnimals = animals.sort(<)
注意每一步的編譯結果都相同,但是最后一步你的閉包里只有一個字符。
問題5 – Swift 1.0 or later
下面的代碼創建了兩個類Address和Person,并且創建了兩個實例對象分別代表Ray和Brain.
class Address {
var fullAddress: String
var city: String
init(fullAddress: String, city: String) {
self.fullAddress = fullAddress
self.city = city
}
}
class Person {
var name: String
var address: Address
init(name: String, address: Address) {
self.name = name
self.address = address
}
}
var headquarters = Address(fullAddress: "123 Tutorial Street", city: "Appletown")
var ray = Person(name: "Ray", address: headquarters)
var brian = Person(name: "Brian", address: headquarters)
假設Brain搬家到街對面的建筑物里,那么你會這樣更新他的地址:
brian.address.fullAddress = "148 Tutorial Street"
這樣做將會發生什么?錯誤出在什么地方呢?
答案:Ray同樣會搬家到新的建筑物里面。Address是一個引用類型類,所以無論你是通過ray或者brain訪問headquarters,訪問都是同一個實例化對象。headquarters對象的變化也會引起ray和brain的變化。你能想象如果Brain收到Ray的郵件或者相反Ray收到Brain的郵件,將會發生什么?解決方案是創建一個新的Address對象賦值給Brain或者把Address聲明成為結構體而不是一個類。
中級
問題1– Swift 2.0 or later
思考下面的代碼:
var optional1: String? = nil
var optional2: String? = .None
答案:兩者沒有什么不同。Optional.None(簡稱.None)是optional變量值初始化的標準方法,而nil只是.None語法的一種修飾。事實上下面語句輸出是正確的:
nil == .None // On Swift 1.x this doesn't compile. You need Optional
.None
記住枚舉類型的Optional下的None:
enum Optional{
case None
case Some(T)
}
問題2-Swift 1.0 or later
下面是thermometer作為類和結構體的例子:
public class ThermometerClass {
private(set) var temperature: Double = 0.0
public func registerTemperature(temperature: Double) {
self.temperature = temperature
}
}
let thermometerClass = ThermometerClass()
thermometerClass.registerTemperature(56.0)
public struct ThermometerStruct {
private(set) var temperature: Double = 0.0
public mutating func registerTemperature(temperature: Double) {
self.temperature = temperature
}
}
let thermometerStruct = ThermometerStruct()
thermometerStruct.registerTemperature(56.0)
但是這段代碼編譯失敗了,請問哪里報錯,出錯的原因是什么。
建議:在使用Playground之前,認真閱讀代碼并思考。
答案:代碼的最后一行不會被編譯通過。ThermometerStruct結構體中正確的聲明了一個mutating屬性函數,它是用來改變結構體內部temperature屬性的值的,但是編譯器不通過的原因是,通過let創建的不可變的registerTemperature結構體調用了registerTemperature函數。
問題3– Swift 1.0 or later
下面的代碼輸出是什么?并說明理由。
var thing = "cars"
let closure = { [thing] in
print("I love \(thing)")
}
thing = "airplanes"
closure()
答案:輸出的是:I love cars。當閉包被聲明的時候,抓捕列表就復制一份thing變量,所以被捕捉的值并沒有改變,即使你給thing賦了一個新值。
如果你要忽視閉包中捕捉列表的值,那么編譯器引用那個值而不是復制。這種情況下,被引用變量的值的變化將會反映到閉包中,正如下面的代碼所示:
var thing = "cars"
let closure = {
print("I love \(thing)")
}
thing = "airplanes"
closure() // Prints "I love airplanes"
問題4– Swift 2.0 or later
下面是一個全局函數,這個函數的功能是計算數組中特殊值得個數。(待校驗)
func countUniques(array: Array) -> Int {
let sorted = array.sort(<)
let initial: (T?, Int) = (.None, 0)
let reduced = sorted.reduce(initial) { ($1, $0.0 == $1 ? $0.1 : $0.1 + 1) }
return reduced.1
}
它使用了< 和==運算符,他們限制著T(占位類型)的實際類型,也就是說T必須遵循Comparable協議。你可以這樣使用它:
countUniques([1, 2, 3, 3]) // result is 3
現在要求你重寫上面的方法作為Array的擴展方法,然后你就可以這樣寫代碼:
[1, 2, 3, 3].countUniques() // should print 3
如何實現?
答案:在Swift 2.0 中,泛類型可以使用類型約束條件被強制擴展。但是假如這個泛類型不滿足這個類型的約束條件,那么這個擴展方法既不可見也無法調用。
所以countUniques全局函數可以作為Array的擴展方法被重寫如下:
extension Array where Element: Comparable {
func countUniques() -> Int {
let sorted = sort(<)
let initial: (Element?, Int) = (.None, 0)
let reduced = sorted.reduce(initial) { ($1, $0.0 == $1 ? $0.1 : $0.1 + 1) }
return reduced.1
}
}
注意:只有元類型實現了Comparable協議新的方法才可以被使用。例如,如果你在全部是UIView對象的數組中調用countUniques,編譯器將會報錯。
import UIKit
let a = [UIView(), UIView()]
a.countUniques() // compiler error here because UIView doesn't implement Comparable
問題5- Swift 2.0 or later
下面一個函數的功能是計算兩個double(optional)類型的數的相除的結果。在執行除法之前,必須提前滿足三個條件:
被除數必須包含nil值
除數必須為包含nil值
除數不能為零
func pide(pidend: Double?, by pisor: Double?) -> Double? {
if pidend == .None {
return .None
}
if pisor == .None {
return .None
}
if pisor == 0 {
return .None
}
return pidend! / pisor!
}
上面的函數可以正常使用,但是會存在兩個問題: