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

 更新时间:2018年03月25日 10:35:35   投稿:daisy  
这篇文章主要给大家介绍了关于Swift如何调用Objective-C的可变参数函数的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用swift具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。

前言

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

总结

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

相关文章

  • swift实现简单的计算器

    swift实现简单的计算器

    这篇文章主要为大家详细介绍了swift实现简单的计算器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • 如何快速用上Swift静态库详解

    如何快速用上Swift静态库详解

    这篇文章主要给大家介绍了关于如何快速用上Swift静态库的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-03-03
  • Swift 指针底层探索分析

    Swift 指针底层探索分析

    这篇文章主要为大家介绍了Swift 指针底层探索分析,主要包括指针的基本使用,以及指针的内存绑定进行详细分析,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • Swift进阶教程Mirror反射示例详解

    Swift进阶教程Mirror反射示例详解

    这篇文章主要为大家介绍了Swift进阶教程Mirror反射示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Swift实现Selection Sort选择排序算法的实例讲解

    Swift实现Selection Sort选择排序算法的实例讲解

    选择排序是一种稳定的排序算法,且实现代码通常比冒泡排序要来的简单,这里我们就来看一下Swift实现Selection Sort选择排序的实例讲解
    2016-07-07
  • Swift 使用 Observe 监测页面滚动的实现方法

    Swift 使用 Observe 监测页面滚动的实现方法

    这篇文章主要介绍了Swift 使用 Observe 监测页面滚动的实现方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-05-05
  • Swift之UITabBarController 导航控制器的自定义

    Swift之UITabBarController 导航控制器的自定义

    本文给大家介绍swift导航控制器之UITabBarController,本文通过代码实例给大家讲解swift导航控制器,导航控制器类继承UITabBarController,代码简单易懂,需要的朋友可以参考下
    2015-10-10
  • Swift中风味各异的类型擦除实例详解

    Swift中风味各异的类型擦除实例详解

    你也许曾听过类型擦除,甚至也使用过标准库提供的类型擦除类型如 AnySequence,下面这篇文章主要给大家介绍了关于Swift中风味各异的类型擦除的相关资料,需要的朋友可以参考下
    2022-04-04
  • 利用Swift实现一个响应式编程库

    利用Swift实现一个响应式编程库

    最近在学习swift,最近有空所以总结一下最近学习的内容,下面这篇文章主要给大家介绍了关于利用Swift实现一个响应式编程库的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面来一起看看吧。
    2017-12-12
  • 使用 Swift Package 插件生成代码的示例详解

    使用 Swift Package 插件生成代码的示例详解

    这篇文章主要介绍了使用 Swift Package 插件生成代码,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-08-08

最新评论