前言
这个问题是一个朋友问我怎么写,一开始我是拒绝的。我想这种东西网上随便 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 一层了……
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对小牛知识库的支持。
问题内容: 我有一个旧的Objective-C项目,我想调用新的Swift函数和对象,我已经创建了文件“ ”和“ ” 对我来说,从Swift调用函数到Objective-C很容易,但是我有一个反向问题。 因此,我创建了一个简单的类“ System.Swift” 现在我尝试按照此处和文件内的文档进行操作 并且已将其导入到Objective-C类中。此时,在我的Objective-C代码的Object
问题内容: 我的目标c文件中有一个函数(比如说MyBlockExecutor类): 现在,我想从Swift调用此函数。这是我的快速电话: 这会使我的应用程序崩溃。我在这条线上收到EXC_BAD_ACCESS错误: 虽然,从Objective-C调用相同的功能非常好。另外,我将MyBlock定义为: 我该如何解决? 编辑:我对目标c函数的更改持开放态度,我只是需要以某种方式将闭包的集合从swift传
本文向大家介绍Swift中的可变参数函数介绍,包括了Swift中的可变参数函数介绍的使用技巧和注意事项,需要的朋友参考一下 可变参数函数指的是可以接受任意多个参数的函数,我们最熟悉的可能就是 NSString 的 -stringWithFormat:方法了。在 Objective-C 中,我们使用这个方法生成字符串的写法是这样的: 这个方法中的参数是可以任意变化的,参数的第一项是需要格式化的字符串
问题内容: 我想使用反射调用Rows.Scan()函数。但是,它需要使用可变数量的指针,但是没有很多源示例。我需要使用反射,因为我计划用Query调用中的值填充一个切片。因此,基本上使用来获取行的长度,然后使用来切片以填充通常使用传递给函数的指针填充的数据点。 基本上像这样的代码: 有人举过一个调用可变参数函数的例子,该函数使用我可以查看的反射来获取指针? 编辑:似乎没有做我想要的示例代码。 结果
主要内容:1. 定义方法,2. 方法声明,3. 调用方法,4. 函数参数函数是一组一起执行任务的语句。 每个Objective-C程序都有一个C函数,也就是函数,所有最简单的程序都可以定义为函数。 可将代码划分为单独的函数。如何在不同的函数之间划分代码取决于程序员,但逻辑上这个划分通常是这样,每个函数执行一个特定的任务。 函数声明告诉编译器函数的名称,返回类型和参数。 函数定义提供函数的实际主体。 在Objective-C中,基本上会将函数称为方法。 Objectiv
问题内容: 有没有办法调用从Swift的dylib加载的C函数? 这是我的 dylib 文件: cppdemofile.cpp cppdemofile.h 编译为dylib并检查: …复制到… Swift程序 : 如何调用该函数?可以使用dylib吗?我应该改为将这些资源添加到我的迅速项目中吗? 问题答案: 从Swift 调用该函数是可能的,因为您已将其定义为与C关联。 使库成为Swift模块(如