Swift如何調(diào)用Objective-C的可變參數(shù)函數(shù)詳解
前言
這個(gè)問(wèn)題是一個(gè)朋友問(wèn)我怎么寫(xiě),一開(kāi)始我是拒絕的。我想這種東西網(wǎng)上隨便 google 下不就有了嗎。他說(shuō),查了,但沒(méi)大看明白。于是我就查了下,沒(méi)想到這個(gè)寫(xiě)法確實(shí)有點(diǎn)詭異,我第一反應(yīng)也沒(méi)看明白。所以隨便水一篇文章,強(qiáng)行完成本周的博客任務(wù),順便給朋友一個(gè)交代。
本文分為兩部分,第一部分是 Swift 怎么調(diào)用 Objective-C 的可變參數(shù)函數(shù),第二部分是 Objective-C 怎么調(diào)用 Swift 的可變參數(shù)函數(shù)。
Swift 調(diào)用 Objective-C 的可變參數(shù)函數(shù)
先寫(xiě)一個(gè)例子
隨便寫(xiě)一個(gè) Objective-C 的可變參數(shù)函數(shù):接受 n 個(gè) String 類(lèi)型的參數(shù),把它們一個(gè)一個(gè)地打印出來(lái),然后返回參數(shù)一共有多少個(gè)。這個(gè)方法毫無(wú)意義,只是為了強(qiáng)行有個(gè)返回值做例子編出來(lái)的而已……
- (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; }
這個(gè)方法直接在 swift 里調(diào)是調(diào)不了的。為了想要在 swift 里調(diào)用,需要把它稍微改造下。
怎么改造一下
把方法簽名里的 ,... 改成一個(gè)參數(shù) args:(va_list)list
va_list list;
和 va_start(list, value);
這兩句需要去掉,因?yàn)槲覀兊?va_list 是傳進(jìn)來(lái)的。 va_end 應(yīng)該也可以去掉了,不去掉也不會(huì)報(bào)錯(cuò),也許也可以保留著作為一個(gè) 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 里怎么調(diào)用
既然 va_list 是作為一個(gè)參數(shù)傳進(jìn)去的,關(guān)鍵是要用特殊方法構(gòu)造一個(gè) va_list 。就跟在 Objective-C 里可以用 malloc 來(lái)強(qiáng)行構(gòu)造 va_list 一樣,Swift 里也有辦法,有一個(gè)函數(shù)可以用:
public func withVaList<R>(_ args: [CVarArg], _ body: (CVaListPointer) -> R) -> R
這個(gè)函數(shù)的形式看起來(lái)不大常見(jiàn),其實(shí)也很簡(jiǎn)單,它就是接受一個(gè)數(shù)組作為第一個(gè)參數(shù),第二個(gè)參數(shù)是個(gè)閉包,閉包的參數(shù)就是生成好的 va_list ,而返回值你隨便返回什么都可以,閉包的返回值就是整個(gè)函數(shù)的返回值。
換句話說(shuō),就是你先傳給它一個(gè)數(shù)組,讓它根據(jù)這個(gè)數(shù)組構(gòu)造 va_list ;然后它把構(gòu)造好的 va_list 用閉包的參數(shù)傳回來(lái)給你,那么在閉包里這個(gè) va_list 就隨你怎么用了;如果閉包里你有什么結(jié)果想傳出去的,可以作為閉包的返回值返回,它就會(huì)作為這個(gè)函數(shù)的返回值傳出去,接受了這個(gè)返回值,后面就隨你怎么用了。
let testClass = TestClass() let count = withVaList(["hello", "hamster", "good", "morning"]) { args -> Int in return testClass.foo(args) } print(count)
輸出:
hello
hamster
good
morning
4
文檔里說(shuō)了,這個(gè)生成的 va_list 只許你在閉包里用,你不許把它傳出去在外面用,不然不保證 valid。讓我們皮一下試試……
let testClass = TestClass() let args = withVaList(["hello", "hamster", "good", "morning"]) { args -> CVaListPointer in return args } print(testClass.foo(args))
結(jié)果是 crash,EXC_BAD_ACCESS,估計(jì)是到了閉包外面那塊空間已經(jīng)被釋放掉了。這也從側(cè)面證明了不需要再寫(xiě) va_end 了吧……
還有另一個(gè)類(lèi)似的函數(shù) getVaList ,把 va_list 作為返回值返回出來(lái)的,寫(xiě)法更簡(jiǎn)潔,把上面的寫(xiě)法改改就是這樣:
let count = testClass.foo(getVaList(["hello", "hamster", "good", "morning"])) print(count)
但是文檔明確說(shuō)了兩點(diǎn):
- 能用 withVaList 就不要用 getVaList 。具體原因沒(méi)說(shuō)。
- 那為啥還要提供給你這個(gè)方法呢?是因?yàn)橛行┣闆r語(yǔ)言規(guī)則不讓用 withVaList ,比如在 class initializer 里。這時(shí)候就只好用 getVaList 了。
包裝成 Swift 的可變參數(shù)方法
上面這語(yǔ)法,如果要用得很多,每次都這么寫(xiě)怪煩的。我們可以給它包裝成一個(gè) Swift 的可變參數(shù)方法……
extension TestClass { func foo(_ strings: String...) -> Int { return withVaList(strings) { args -> Int in return foo(args) } } }
然后調(diào)用的時(shí)候就一勞永逸了:
let testClass = TestClass() let count = testClass.foo("hello", "hamster", "good", "morning") print(count)
感慨下 Swift 的語(yǔ)法簡(jiǎn)潔太多了,不是嗎?
Objective-C 調(diào)用 Swift 的可變參數(shù)函數(shù)
既然 Swift 的語(yǔ)法這么簡(jiǎn)潔,我們干脆把可變參數(shù)方法都在 Swift 里實(shí)現(xiàn),然后讓 Objective-C 來(lái)調(diào)唄?
然而 Swift 無(wú)情地拒絕了:
真的要調(diào)怎么辦?只好另寫(xiě)一個(gè)接受數(shù)組為參數(shù)的方法,在 Objective-C 里調(diào)這個(gè)方法,或者再寫(xiě)一個(gè) Objective-C 的可變參數(shù)方法把它 wrap 一層了……
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)我們的支持。
上一篇:Swift3遷移至Swift4可能遇到的問(wèn)題小結(jié)
欄 目:Swift
下一篇:swift4.0實(shí)現(xiàn)視頻 、屏幕旋轉(zhuǎn)、倍速 、手勢(shì)調(diào)節(jié)及鎖屏面板等功能實(shí)例
本文標(biāo)題:Swift如何調(diào)用Objective-C的可變參數(shù)函數(shù)詳解
本文地址:http://mengdiqiu.com.cn/a1/Swift/11950.html
您可能感興趣的文章
- 01-11swift中defer幾個(gè)簡(jiǎn)單的使用場(chǎng)景詳解
- 01-11Swift利用Decodable解析JSON的一個(gè)小問(wèn)題詳解
- 01-11Swift中defer關(guān)鍵字推遲執(zhí)行示例詳解
- 01-11Swift中初始化init的方法小結(jié)
- 01-11Swift中定義單例的方法實(shí)例
- 01-11Swift利用純代碼實(shí)現(xiàn)時(shí)鐘效果實(shí)例代碼
- 01-11Swift中排序算法的簡(jiǎn)單取舍詳解
- 01-11Swift如何為設(shè)置中心添加常用功能
- 01-11Swift Json實(shí)例詳細(xì)解析
- 01-11Swift利用指紋識(shí)別或面部識(shí)別為應(yīng)用添加私密保護(hù)功能


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-11Swift利用Decodable解析JSON的一個(gè)小問(wèn)題
- 01-11swift中defer幾個(gè)簡(jiǎn)單的使用場(chǎng)景詳解
- 01-11Swift中初始化init的方法小結(jié)
- 01-11Swift中defer關(guān)鍵字推遲執(zhí)行示例詳解
- 01-11Swift利用純代碼實(shí)現(xiàn)時(shí)鐘效果實(shí)例代碼
- 01-11Swift中定義單例的方法實(shí)例
- 01-11Swift中排序算法的簡(jiǎn)單取舍詳解
- 01-11Swift Json實(shí)例詳細(xì)解析
- 01-11Swift如何為設(shè)置中心添加常用功能
- 01-11Swift利用指紋識(shí)別或面部識(shí)別為應(yīng)用添
隨機(jī)閱讀
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子