答《卓同學(xué)的 Swift 面試題》上

原文鏈接 卓同學(xué)的 Swift 面試題
下篇http://www.reibang.com/p/cc4a737ddc1d

class 和 struct 的區(qū)別

class 為類, struct 為結(jié)構(gòu)體, 類是引用類型, 結(jié)構(gòu)體為值類型, 結(jié)構(gòu)體不可以繼承

不通過繼承移必,代碼復(fù)用(共享)的方式有哪些

擴(kuò)展, 全局函數(shù)

Set 獨(dú)有的方法有哪些勾缭?

// 定義一個 set
let setA: Set<Int> = [1, 2, 3, 4, 4]// {1, 2, 3, 4}, 順序可能不一致, 同一個元素只有一個值
let setB: Set<Int> = [1, 3, 5, 7, 9]// {1, 3, 5, 7, 9}
// 取并集 A | B
let setUnion = setA.union(setB)// {1, 2, 3, 4, 5, 7, 9}
// 取交集 A & B
let setIntersect = setA.intersection(setB)// {1, 3}
// 取差集 A - B
let setRevers = setA.subtracting(setB) // {2, 4}
// 取對稱差集, A XOR B = A - B | B - A
let setXor = setA.symmetricDifference(setB) //{2, 4, 5, 7, 9}
A B 集合

A B 并集, A | B

A B 交集 A & B

A B差集 A - B

A B對稱差集 A XOR B

實(shí)現(xiàn)一個 min 函數(shù)狂男,返回兩個元素較小的元素

func myMin<T: Comparable>(_ a: T, _ b: T) -> T {
    return a < b ? a : b
}
myMin(1, 2)

map、filter挂滓、reduce 的作用

map 用于映射, 可以將一個列表轉(zhuǎn)換為另一個列表

[1, 2, 3].map{"\($0)"}// 數(shù)字?jǐn)?shù)組轉(zhuǎn)換為字符串?dāng)?shù)組
["1", "2", "3"]

filter 用于過濾, 可以篩選出想要的元素

[1, 2, 3].filter{$0 % 2 == 0} // 篩選偶數(shù)
// [2]

reduce 合并

[1, 2, 3].reduce(""){$0 + "\($1)"}// 轉(zhuǎn)換為字符串并拼接
// "123"

組合示例

(0 ..< 10).filter{$0 % 2 == 0}.map{"\($0)"}.reduce(""){$0 + $1}
// 02468

map 與 flatmap 的區(qū)別

flatmap 有兩個實(shí)現(xiàn)函數(shù)實(shí)現(xiàn),
public func flatMap<ElementOfResult>(_ transform: (Element) throws -> ElementOfResult?) rethrows -> [ElementOfResult]
這個方法, 中間的函數(shù)返回值為一個可選值, 而 flatmap 會丟掉那些返回值為 nil 的值
例如

["1", "@", "2", "3", "a"].flatMap{Int($0)}
// [1, 2, 3]
["1", "@", "2", "3", "a"].map{Int($0) ?? -1}
//[Optional(1), nil, Optional(2), Optional(3), nil]

另一個實(shí)現(xiàn)
public func flatMap<SegmentOfResult>(_ transform: (Element) throws -> SegmentOfResult) rethrows -> [SegmentOfResult.Iterator.Element] where SegmentOfResult : Sequence
中間的函數(shù), 返回值為一個數(shù)組, 而這個 flapmap 返回的對象則是一個與自己元素類型相同的數(shù)組

func someFunc(_ array:[Int]) -> [Int] {
    return array
}
[[1], [2, 3], [4, 5, 6]].map(someFunc)
// [[1], [2, 3], [4, 5, 6]]
[[1], [2, 3], [4, 5, 6]].flatMap(someFunc)
// [1, 2, 3, 4, 5, 6]

其實(shí)這個實(shí)現(xiàn), 相當(dāng)于是在使用 map 之后, 再將各個數(shù)組拼起來一樣的

[[1], [2, 3], [4, 5, 6]].map(someFunc).reduce([Int]()) {$0 + $1}
// [1, 2, 3, 4, 5, 6]

什么是 copy on write時候

寫時復(fù)制, 指的是 swift 中的值類型, 并不會在一開始賦值的時候就去復(fù)制, 只有在需要修改的時候, 才去復(fù)制.
這里有詳細(xì)的說明
http://www.reibang.com/p/7e8ba0659646

如何獲取當(dāng)前代碼的函數(shù)名和行號

#file 用于獲取當(dāng)前文件文件名
#line 用于獲取當(dāng)前行號
#column 用于獲取當(dāng)前列編號
#function 用于獲取當(dāng)前函數(shù)名
以上這些都是特殊的字面量, 多用于調(diào)試輸出日志
具體可以看這里 apple 文檔
https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/Expressions.html
這里有中文翻譯
http://wiki.jikexueyuan.com/project/swift/chapter3/04_Expressions.html

如何聲明一個只能被類 conform 的 protocol

聲明協(xié)議的時候, 加一個 class 即可

protocol SomeClassProtocl: class {
    func someFunction()
}

guard 使用場景

guard 和 if 類似, 不同的是, guard 總是有一個 else 語句, 如果表達(dá)式是假或者值綁定失敗的時候, 會執(zhí)行 else 語句, 且在 else 語句中一定要停止函數(shù)調(diào)用
例如

guard 1 + 1 == 2 else {
    fatalError("something wrong")
}

常用使用場景為, 用戶登錄的時候, 驗(yàn)證用戶是否有輸入用戶名密碼等

guard let userName = self.userNameTextField.text,
  let password = self.passwordTextField.text else {
    return
}

defer 使用場景

defer 語句塊中的代碼, 會在當(dāng)前作用域結(jié)束前調(diào)用, 常用場景如異常退出后, 關(guān)閉數(shù)據(jù)庫連接

func someQuery() -> ([Result], [Result]){
    let db = DBOpen("xxx")
    defer {
        db.close()
    }
    guard results1 = db.query("query1") else {
        return nil
    }
    guard results2 = db.query("query2") else {
        return nil
    }
    return (results1, results2)
}

需要注意的是, 如果有多個 defer, 那么后加入的先執(zhí)行

func someDeferFunction() {
    defer {
        print("\(#function)-end-1-1")
        print("\(#function)-end-1-2")
    }
    defer {
        print("\(#function)-end-2-1")
        print("\(#function)-end-2-2")
    }
    if true {
        defer {
            print("if defer")
        }
        print("if end")
    }
    print("function end")
}
someDeferFunction()
// 輸出
// if end
// if defer
// function end
// someDeferFunction()-end-2-1
// someDeferFunction()-end-2-2
// someDeferFunction()-end-1-1
// someDeferFunction()-end-1-2

String 與 NSString 的關(guān)系與區(qū)別

NSString 與 String 之間可以隨意轉(zhuǎn)換,

let someString = "123"
let someNSString = NSString(string: "n123")
let strintToNSString = someString as NSString
let nsstringToString = someNSString as String

String 是結(jié)構(gòu)體, 值類型, NSString 是類, 引用類型.
通常, 沒必要使用 NSString 類, 除非你要使用一些特有方法, 例如使用 pathExtension 屬性

怎么獲取一個 String 的長度

不考慮編碼, 只是想知道字符的數(shù)量, 用characters.count

"hello".characters.count // 5
"你好".characters.count // 2
"こんにちは".characters.count // 5

如果想知道在某個編碼下占多少字節(jié), 可以用

"hello".lengthOfBytes(using: .ascii) // 5
"hello".lengthOfBytes(using: .unicode) // 10
"你好".lengthOfBytes(using: .unicode) // 4
"你好".lengthOfBytes(using: .utf8) // 6
"こんにちは".lengthOfBytes(using: .unicode) // 10
"こんにちは".lengthOfBytes(using: .utf8) // 15

如何截取 String 的某段字符串

swift 中, 有三個取子串函數(shù),
substring:to , substring:from, substring:with.

let simpleString = "Hello, world"
simpleString.substring(to: simpleString.index(simpleString.startIndex, offsetBy: 5))
// hello
simpleString.substring(from: simpleString.index(simpleString.endIndex, offsetBy: -5))
// world
simpleString.substring(with: simpleString.index(simpleString.startIndex, offsetBy: 5) ..< simpleString.index(simpleString.endIndex, offsetBy: -5))
// ,

使用起來略微麻煩, 具體用法可以參考我的另一篇文章http://www.reibang.com/p/b3231f9406e9

throws 和 rethrows 的用法與作用

throws 用在函數(shù)上, 表示這個函數(shù)會拋出錯誤.
有兩種情況會拋出錯誤, 一種是直接使用 throw 拋出, 另一種是調(diào)用其他拋出異常的函數(shù)時, 直接使用 try xx 沒有處理異常.

enum DivideError: Error {
    case EqualZeroError;
}
func divide(_ a: Double, _ b: Double) throws -> Double {
    guard b != Double(0) else {
        throw DivideError.EqualZeroError
    }
    return a / b
}
func split(pieces: Int) throws -> Double {
    return try divide(1, Double(pieces))
}

rethrows 與 throws 類似, 不過只適用于參數(shù)中有函數(shù), 且函數(shù)會拋出異常的情況, rethrows 可以用 throws 替換, 反過來不行

func processNumber(a: Double, b: Double, function: (Double, Double) throws -> Double) rethrows -> Double {
    return try function(a, b)
}

try? 和 try啸胧!是什么意思

這兩個都用于處理可拋出異常的函數(shù), 使用這兩個關(guān)鍵字可以不用寫 do catch.
區(qū)別在于, try? 在用于處理可拋出異常函數(shù)時, 如果函數(shù)拋出異常, 則返回 nil, 否則返回函數(shù)返回值的可選值, 如:

print(try? divide(2, 1))
// Optional(2.0)
print(try? divide(2, 0))
// nil

而 try! 則在函數(shù)拋出異常的時候崩潰, 否則則返會函數(shù)返回值, 相當(dāng)于(try? xxx)!, 如:

print(try! divide(2, 1))
// 2.0
print(try! divide(2, 0))
// 崩潰

associatedtype 的作用

簡單來說就是 protocol 使用的泛型
例如定義一個列表協(xié)議

protocol ListProtcol {
    associatedtype Element
    func push(_ element:Element)
    func pop(_ element:Element) -> Element?
}

實(shí)現(xiàn)協(xié)議的時候, 可以使用 typealias 指定為特定的類型, 也可以自動推斷, 如

class IntList: ListProtcol {
    typealias Element = Int // 使用 typealias 指定為 Int
    var list = [Element]()
    func push(_ element: Element) {
        self.list.append(element)
    }
    func pop(_ element: Element) -> Element? {
        return self.list.popLast()
    }
}
class DoubleList: ListProtcol {
    var list = [Double]()
    func push(_ element: Double) {// 自動推斷
        self.list.append(element)
    }
    func pop(_ element: Double) -> Double? {
        return self.list.popLast()
    }
}

使用泛型也可以

class AnyList<T>: ListProtcol {
    var list = [T]()
    func push(_ element: T) {
        self.list.append(element)
    }
    func pop(_ element: T) -> T? {
        return self.list.popLast()
    }
}

可以使用 where 字句限定 Element 類型, 如:

extension ListProtcol where Element == Int {
    func isInt() ->Bool {
        return true
    }
}

什么時候使用 final

final 用于限制繼承和重寫. 如果只是需要在某一個屬性前加一個 final.
如果需要限制整個類無法被繼承, 那么可以在類名之前加一個final

public 和 open 的區(qū)別

這兩個都用于在模塊中聲明需要對外界暴露的函數(shù), 區(qū)別在于, public 修飾的類, 在模塊外無法繼承, 而 open 則可以任意繼承, 公開度來說, public < open

聲明一個只有一個參數(shù)沒有返回值閉包的別名

沒有返回值也就是返回值為 Void

typealias SomeClosuerType = (String) -> (Void)
let someClosuer: SomeClosuerType = { (name: String) in
    print("hello,", name)
}
someClosuer("world")
// hello, world

Self 的使用場景

Self 通常在協(xié)議中使用, 用來表示實(shí)現(xiàn)者或者實(shí)現(xiàn)者的子類類型.
例如, 定義一個復(fù)制的協(xié)議

protocol CopyProtocol {
    func copy() -> Self
}

如果是結(jié)構(gòu)體去實(shí)現(xiàn), 要將Self 換為具體的類型

struct SomeStruct: CopyProtocol {
    let value: Int
    func copySelf() -> SomeStruct {
        return SomeStruct(value: self.value)
    }
}

如果是類去實(shí)現(xiàn), 則有點(diǎn)復(fù)雜, 需要有一個 required 初始化方法, 具體可以看這里 http://swifter.tips/use-self/

class SomeCopyableClass: CopyProtocol {
    func copySelf() -> Self {
        return type(of: self).init()
    }
    required init(){}
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末赶站,一起剝皮案震驚了整個濱河市幔虏,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌贝椿,老刑警劉巖想括,帶你破解...
    沈念sama閱讀 217,406評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異烙博,居然都是意外死亡瑟蜈,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評論 3 393
  • 文/潘曉璐 我一進(jìn)店門渣窜,熙熙樓的掌柜王于貴愁眉苦臉地迎上來铺根,“玉大人,你說我怎么就攤上這事乔宿∥挥兀” “怎么了?”我有些...
    開封第一講書人閱讀 163,711評論 0 353
  • 文/不壞的土叔 我叫張陵予颤,是天一觀的道長囤官。 經(jīng)常有香客問我,道長蛤虐,這世上最難降的妖魔是什么党饮? 我笑而不...
    開封第一講書人閱讀 58,380評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮驳庭,結(jié)果婚禮上刑顺,老公的妹妹穿的比我還像新娘。我一直安慰自己饲常,他們只是感情好蹲堂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,432評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著贝淤,像睡著了一般柒竞。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上播聪,一...
    開封第一講書人閱讀 51,301評論 1 301
  • 那天朽基,我揣著相機(jī)與錄音,去河邊找鬼离陶。 笑死稼虎,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的招刨。 我是一名探鬼主播霎俩,決...
    沈念sama閱讀 40,145評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了打却?” 一聲冷哼從身側(cè)響起杉适,我...
    開封第一講書人閱讀 39,008評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎学密,沒想到半個月后淘衙,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,443評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡腻暮,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,649評論 3 334
  • 正文 我和宋清朗相戀三年彤守,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片哭靖。...
    茶點(diǎn)故事閱讀 39,795評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡具垫,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出试幽,到底是詐尸還是另有隱情筝蚕,我是刑警寧澤,帶...
    沈念sama閱讀 35,501評論 5 345
  • 正文 年R本政府宣布铺坞,位于F島的核電站起宽,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏济榨。R本人自食惡果不足惜坯沪,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,119評論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望擒滑。 院中可真熱鬧腐晾,春花似錦、人聲如沸丐一。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽库车。三九已至巨柒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間柠衍,已是汗流浹背洋满。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留拧略,地道東北人。 一個月前我還...
    沈念sama閱讀 47,899評論 2 370
  • 正文 我出身青樓瘪弓,卻偏偏與公主長得像垫蛆,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,724評論 2 354

推薦閱讀更多精彩內(nèi)容

  • 題目: 卓同學(xué)的 Swift 面試題 翻了 喵神 和 SwiftGG 的資料, 借鑒了 Arco_vv 和 En...
    青木KON閱讀 1,948評論 3 8
  • 原題的鏈接:http://www.reibang.com/p/7c7f4b4e4efe 1. class 和 st...
    Swifter丶閱讀 10,235評論 5 46
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法袱饭,類相關(guān)的語法川无,內(nèi)部類的語法,繼承相關(guān)的語法虑乖,異常的語法懦趋,線程的語...
    子非魚_t_閱讀 31,630評論 18 399
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)疹味,斷路器仅叫,智...
    卡卡羅2017閱讀 134,654評論 18 139
  • 以前我讀書不會拼音,落字糙捺。然后诫咱,我現(xiàn)在讀書好啦,百步穿楊都能完整的讀下來了洪灯。以前我肚子里沒有墨水坎缭,所以我讀的不好。...
    大皓皓啊閱讀 182評論 0 0