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

前言

这个问题是一个朋友问我怎么写,一开始我是拒绝的。我想这种东西网上随便 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)

但是文档明确说了两点:

  • 能用 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 无情地拒绝了:

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

总结

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

(0)

相关推荐

  • 在一个项目中同时使用Swift和Objective-C代码混合编程的方法

    Swift 与 Objective-C 的兼容能力使你可以在同一个工程中同时使用两种语言.你可以用这种叫做 mix and match 的特性来开发基于混合语言的应用,可以用 Swfit 的最新特性实现应用的一部分功能,并无缝地并入已有的 Objective-C 的代码中. Mix and Match 概述 Objective-C 和 Swift 文件可以在一个工程中并存,不管这个工程原本是基于 Objective-C 还是 Swift.你可以直接往现有工程中简单地添加另一种语言的源文件.这种自

  • Objective-c代码如何移植为Swift代码 Objective-c代码转移到Swift过程介绍

    迁移工作正好提供了一个重新审视现有 Objective-C 应用程序的机会,也可以通过 Swift 代码来更好的优化应用程序的体系架构,逻辑以及性能.直接的说,你将用先前学到的 mix and match 以及这两个语言间的互操作性来进行增量迁移工作.Mix-and-match 功能使得选择哪些特性和功能来用 Swift 来实现,哪些依然用 Objective-C 来实现变得简单.Swift 和 Objective-C 的互用性又使得将这些功能集成到 Objective-C 变得并不困难.通过这

  • Swift和Objective-C 混编注意事项

    Swift和Objective-C 混编注意事项整理: 前言 Swift已推出数年,与Objective-C相比Swift的语言机制及使用简易程度上更接地气,大大降低了iOS入门门槛.当然这对新入行的童鞋没来讲,的确算是福音,但对于整个iOS编程从业者来讲,真真是,曾几何时"高大上",转瞬之间"矮矬穷".再加上培训班横行,批量批发之下,iOS再也看不到当年的辉煌.iOS10推出后,紧跟着Xcode8也推送了更新,细心者会发现,Xcode8下iOS版本最低适配已变为i

  • 在Swift中使用Objective-C编写类、继承Objective-C类

    互用性(互操作性)使开发者可以定义融合了 Objective-C 语言特性的Swift类.编写 Swift 类时,不仅可以继承 Objective-C 语言编写的父类,采用 Objective-C 的协议,还可以利用 Objective-C 的一些其它功能.这意味着,开发者可以基于 Objective-C 中已有的熟悉.可靠的类.方法和框架来创建 Swift 类,并结合 Swift 提供的现代化和更有效的语言特点对其进行优化. 继承Objective-C的类 在 Swift 中,开发者可以定义一

  • Swift能代替Objective-C吗?

    我文章的中心是,以 Apple 目前给出的各种资料来看,这语言不会替代掉ObjC,它不是下一代的ObjC.它有很多缺点,使得它不足以做大型项目.这些缺点使得,Apple 自己都没有使用它做 Mac/iOS 的 app.我不排除明年后年它有很大改进,但至少现在还没有这端倪. 因此,如果你会ObjC,你不需要去看它. 但你如果问我这语言对普通开发者重要不重要,我说重要,可以明确告诉你这一点--它是 Apple 在 WWDC 向全世界推出的重磅语言我怎麽能说它不重要? 它降低了入门的门槛.使得大量的

  • Swift调用Objective-C代码

    最近iOS开发之新编程语言Swift在iOS开发圈内反响比较大,国内外都有很多教程或小示例. 虽然Swift这门语言仍然在不断的进化之中,而且变动还是比较大,苹果公司也不承诺目前所写代码会在将来兼容,但仍挡不住iOS开发者的热情. 为什么要使用Swift调用Objective-C代码 目前Swift语言所编写的应用才刚刚可以使用Xcode 6 GM版本提交,而Objective-C作为苹果的主开发语言存在了很多年了.目前尚无成熟的Swift库可用,所以当前编写应用可以说基本离不开调用Object

  • Objective-C和Swift的转换速查手册(推荐)

    前言 如果你正要从Objective-C过渡到Swift,或反过来,一个在两种语言间显示等效代码的小手册会很有帮助.本文内容就是这些:苹果开发者的红宝书,包含变量,集合,函数,类等等. 下面例子中,上面是Objective-C代码,下面是等效的Swift代码.必要的地方我会给一些备注来帮助你理解. 变量与常量 创建一个变量 //Objective-C NSInteger score = 556; // NSString *name = @"Taylor"; // BOOL logged

  • Swift调用Objective-C编写的API实例

    互用性是让 Swift 和 Objective-C 相接合的一种特性,使你能够在一种语言编写的文件中使用另一种语言.当你准备开始把 Swift 融入到你的开发流程中时,你应该懂得如何利用互用性来重新定义并提高你写 Cocoa 应用的方案. 互用性很重要的一点就是允许你在写 Swift 代码时使用 Objective-C 的 API 接口.当你导入一个 Objective-C 框架后,你可以使用原生的 Swift 语法实例化它的 Class 并且与之交互. 初始化 为了使用 Swift 实例化 O

  • Objective-C中的block与Swift中的尾随闭包使用教程

    前言 在项目开发中经常会去查iOS闭包怎么写,因为它的语法太古怪,两种语言写法不一,经常搞混,干脆记录下常用的写法算了 闭包定义 闭包是指可以包含自由(未绑定到特定对象)变量的代码块:这些变量不是在这个代码块内或者任何全局上下文中定义的,而是在定义代码块的环境中定义(局部变量)."闭包" 一词来源于以下两者的结合:要执行的代码块(由于自由变量被包含在代码块中,这些自由变量以及它们引用的对象没有被释放)和为自由变量提供绑定的计算环境(作用域). OC中的block与Swift中的尾随闭包

  • Swift、Objective-C、Cocoa混合编程设置指南

    Swift 被设计用来无缝兼容 Cocoa 和 Objective-C .在 Swift 中,你可以使用 Objective-C 的 API(包括系统框架和你自定义的代码),你也可以在 Objective-C中 使用 Swift 的 API.这种兼容性使 Swift 变成了一个简单.方便并且强大的工具集成到你的 Cocoa 应用开发工作流程中. 这篇指南包括了三个有关兼容性的重要方面方便你更好地利用来开发 Cocoa 应用: 互用性 使你将 Swift 和 Objective-C 相接合,允许在

随机推荐