国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Swift > 正文

Swift如何調(diào)用Objective-C的可變參數(shù)函數(shù)詳解

2020-03-09 17:42:07
字體:
供稿:網(wǎng)友

前言

這個問題是一個朋友問我怎么寫,一開始我是拒絕的。我想這種東西網(wǎng)上隨便 google 下不就有了嗎。他說,查了,但沒大看明白。于是我就查了下,沒想到這個寫法確實(shí)有點(diǎn)詭異,我第一反應(yīng)也沒看明白。所以隨便水一篇文章,強(qiáng)行完成本周的博客任務(wù),順便給朋友一個交代。

本文分為兩部分,第一部分是 Swift 怎么調(diào)用 Objective-C 的可變參數(shù)函數(shù),第二部分是 Objective-C 怎么調(diào)用 Swift 的可變參數(shù)函數(shù)。

Swift 調(diào)用 Objective-C 的可變參數(shù)函數(shù)

先寫一個例子

隨便寫一個 Objective-C 的可變參數(shù)函數(shù):接受 n 個 String 類型的參數(shù),把它們一個一個地打印出來,然后返回參數(shù)一共有多少個。這個方法毫無意義,只是為了強(qiáng)行有個返回值做例子編出來的而已……

- (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 里調(diào)是調(diào)不了的。為了想要在 swift 里調(diào)用,需要把它稍微改造下。

怎么改造一下

把方法簽名里的 ,... 改成一個參數(shù) args:(va_list)list

va_list list;va_start(list, value); 這兩句需要去掉,因?yàn)槲覀兊?va_list 是傳進(jìn)來的。 va_end 應(yīng)該也可以去掉了,不去掉也不會報錯,也許也可以保留著作為一個 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 是作為一個參數(shù)傳進(jìn)去的,關(guān)鍵是要用特殊方法構(gòu)造一個 va_list 。就跟在 Objective-C 里可以用 malloc 來強(qiáng)行構(gòu)造 va_list 一樣,Swift 里也有辦法,有一個函數(shù)可以用:

public func withVaList<R>(_ args: [CVarArg], _ body: (CVaListPointer) -> R) -> R

這個函數(shù)的形式看起來不大常見,其實(shí)也很簡單,它就是接受一個數(shù)組作為第一個參數(shù),第二個參數(shù)是個閉包,閉包的參數(shù)就是生成好的 va_list ,而返回值你隨便返回什么都可以,閉包的返回值就是整個函數(shù)的返回值。

換句話說,就是你先傳給它一個數(shù)組,讓它根據(jù)這個數(shù)組構(gòu)造 va_list ;然后它把構(gòu)造好的 va_list 用閉包的參數(shù)傳回來給你,那么在閉包里這個 va_list 就隨你怎么用了;如果閉包里你有什么結(jié)果想傳出去的,可以作為閉包的返回值返回,它就會作為這個函數(shù)的返回值傳出去,接受了這個返回值,后面就隨你怎么用了。

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))

結(jié)果是 crash,EXC_BAD_ACCESS,估計是到了閉包外面那塊空間已經(jīng)被釋放掉了。這也從側(cè)面證明了不需要再寫 va_end 了吧……

還有另一個類似的函數(shù) getVaList ,把 va_list 作為返回值返回出來的,寫法更簡潔,把上面的寫法改改就是這樣:

let count = testClass.foo(getVaList(["hello", "hamster", "good", "morning"]))print(count)

但是文檔明確說了兩點(diǎn):

  • 能用 withVaList 就不要用 getVaList 。具體原因沒說。
  • 那為啥還要提供給你這個方法呢?是因?yàn)橛行┣闆r語言規(guī)則不讓用 withVaList ,比如在 class initializer 里。這時候就只好用 getVaList 了。

包裝成 Swift 的可變參數(shù)方法

上面這語法,如果要用得很多,每次都這么寫怪煩的。我們可以給它包裝成一個 Swift 的可變參數(shù)方法……

extension TestClass { func foo(_ strings: String...) -> Int { return withVaList(strings) { args -> Int in  return foo(args) } }}

然后調(diào)用的時候就一勞永逸了:

let testClass = TestClass()let count = testClass.foo("hello", "hamster", "good", "morning")print(count)

感慨下 Swift 的語法簡潔太多了,不是嗎?

Objective-C 調(diào)用 Swift 的可變參數(shù)函數(shù)

既然 Swift 的語法這么簡潔,我們干脆把可變參數(shù)方法都在 Swift 里實(shí)現(xiàn),然后讓 Objective-C 來調(diào)唄?

然而 Swift 無情地拒絕了:

swift,objectivec,函數(shù)調(diào)用

真的要調(diào)怎么辦?只好另寫一個接受數(shù)組為參數(shù)的方法,在 Objective-C 里調(diào)這個方法,或者再寫一個 Objective-C 的可變參數(shù)方法把它 wrap 一層了……

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對VEVB武林網(wǎng)的支持。


注:相關(guān)教程知識閱讀請移步到swift教程頻道。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 时尚| 新疆| 崇仁县| 河间市| 连城县| 洛宁县| 德惠市| 昭苏县| 东宁县| 阿坝县| 河北省| 讷河市| 阿瓦提县| 阿鲁科尔沁旗| 清涧县| 泸水县| 鄂托克旗| 吕梁市| 麻阳| 宿州市| 微博| 垫江县| 静安区| 香港 | 屏东县| 行唐县| 阿拉善右旗| 开鲁县| 文昌市| 天祝| 姜堰市| 韶关市| 贵南县| 娄烦县| 乌拉特中旗| 淮阳县| 三河市| 龙山县| 宁蒗| 社会| 宜兰市|