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

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

前言

这个问题是一个朋友问我怎么写,一开始我是拒绝的。我想这种东西网上随便 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 就随你怎么用了;如果闭包里你有什么结果想传出去的,可以作为闭包的返回值返回,它就会作为这个函数的返回值传出去,接受了这个返回值,后面就随你怎么用了。