Swift如何调用Objective-C的可变参数函数详解

2020-01-09 00:08:29王冬梅


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)

但是文档明确说了两点:

能用 withVaList 就不要用 getVaList 。具体原因没说。 那为啥还要提供给你这个方法呢?是因为有些情况语言规则不让用 withVaList ,比如在 class initializer 里。这时候就只好用 getVaList 了。

包装成 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 无情地拒绝了:

swift,objectivec,函数调用

真的要调怎么办?只好另写一个接受数组为参数的方法,在 Objective-C 里调这个方法,或者再写一个 Objective-C 的可变参数方法把它 wrap 一层了……

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对ASPKU的支持。