前言
這個問題是一個朋友問我怎么寫,一開始我是拒絕的。我想這種東西網上隨便 google 下不就有了嗎。他說,查了,但沒大看明白。于是我就查了下,沒想到這個寫法確實有點詭異,我第一反應也沒看明白。所以隨便水一篇文章,強行完成本周的博客任務,順便給朋友一個交代。
本文分為兩部分,第一部分是 Swift 怎么調用 Objective-C 的可變參數函數,第二部分是 Objective-C 怎么調用 Swift 的可變參數函數。
Swift 調用 Objective-C 的可變參數函數
先寫一個例子
隨便寫一個 Objective-C 的可變參數函數:接受 n 個 String 類型的參數,把它們一個一個地打印出來,然后返回參數一共有多少個。這個方法毫無意義,只是為了強行有個返回值做例子編出來的而已……
- (NSInteger)foo:(NSString *)value,...{ va_list list; va_start(list, value); NSInteger count = 0; while (YES) { NSString *string = va_arg(list, NSString*); if (!string) { break; } NSLog(@"%@",string); count++; } va_end(list); return count;}
這個方法直接在 swift 里調是調不了的。為了想要在 swift 里調用,需要把它稍微改造下。
怎么改造一下
把方法簽名里的 ,... 改成一個參數 args:(va_list)list
va_list list;
和 va_start(list, value);
這兩句需要去掉,因為我們的 va_list 是傳進來的。 va_end 應該也可以去掉了,不去掉也不會報錯,也許也可以保留著作為一個 good practice 吧。
改完之后的 Objective-C 方法:
- (NSInteger)foo:(va_list)list{ NSInteger count = 0; while (YES) { NSString *string = va_arg(list, NSString*); if (!string) { break; } NSLog(@"%@",string); count++; } return count;}
在 Swift 里怎么調用
既然 va_list 是作為一個參數傳進去的,關鍵是要用特殊方法構造一個 va_list 。就跟在 Objective-C 里可以用 malloc 來強行構造 va_list 一樣,Swift 里也有辦法,有一個函數可以用:
public func withVaList<R>(_ args: [CVarArg], _ body: (CVaListPointer) -> R) -> R
這個函數的形式看起來不大常見,其實也很簡單,它就是接受一個數組作為第一個參數,第二個參數是個閉包,閉包的參數就是生成好的 va_list ,而返回值你隨便返回什么都可以,閉包的返回值就是整個函數的返回值。
換句話說,就是你先傳給它一個數組,讓它根據這個數組構造 va_list ;然后它把構造好的 va_list 用閉包的參數傳回來給你,那么在閉包里這個 va_list 就隨你怎么用了;如果閉包里你有什么結果想傳出去的,可以作為閉包的返回值返回,它就會作為這個函數的返回值傳出去,接受了這個返回值,后面就隨你怎么用了。
let testClass = TestClass()let count = withVaList(["hello", "hamster", "good", "morning"]) { args -> Int in return testClass.foo(args)}print(count)
輸出:
hello
hamster
good
morning
4
文檔里說了,這個生成的 va_list 只許你在閉包里用,你不許把它傳出去在外面用,不然不保證 valid。讓我們皮一下試試……
let testClass = TestClass()let args = withVaList(["hello", "hamster", "good", "morning"]) { args -> CVaListPointer in return args}print(testClass.foo(args))
結果是 crash,EXC_BAD_ACCESS,估計是到了閉包外面那塊空間已經被釋放掉了。這也從側面證明了不需要再寫 va_end 了吧……
還有另一個類似的函數 getVaList ,把 va_list 作為返回值返回出來的,寫法更簡潔,把上面的寫法改改就是這樣:
let count = testClass.foo(getVaList(["hello", "hamster", "good", "morning"]))print(count)
但是文檔明確說了兩點:
包裝成 Swift 的可變參數方法
上面這語法,如果要用得很多,每次都這么寫怪煩的。我們可以給它包裝成一個 Swift 的可變參數方法……
extension TestClass { func foo(_ strings: String...) -> Int { return withVaList(strings) { args -> Int in return foo(args) } }}
然后調用的時候就一勞永逸了:
let testClass = TestClass()let count = testClass.foo("hello", "hamster", "good", "morning")print(count)
感慨下 Swift 的語法簡潔太多了,不是嗎?
Objective-C 調用 Swift 的可變參數函數
既然 Swift 的語法這么簡潔,我們干脆把可變參數方法都在 Swift 里實現,然后讓 Objective-C 來調唄?
然而 Swift 無情地拒絕了:
真的要調怎么辦?只好另寫一個接受數組為參數的方法,在 Objective-C 里調這個方法,或者再寫一個 Objective-C 的可變參數方法把它 wrap 一層了……
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網的支持。
新聞熱點
疑難解答