浅谈 Swift 2 中的 Objective-C 指针

2015-09-07  499

作者:Jameson Quave,原文链接,原文日期:2015/08/23
译者:mmoaay;校对:numbbbbb;定稿:shanks

本文写于 2015 年 8 月 23 日,与 Xcode7 Beta 版和 Swift 2 兼容

在 Swift 中读 C 指针

下面这个 Objective-C 方法会返回一个 int 指针,或者说 C 术语里面的 (int *)

  1. 1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
  1. @interface PointerBridge : NSObject {
    int count;
    }
    - (int *) getCountPtr;
    @end
  2.  
  3. @implementation PointerBridge
    - (instancetype) init {
    self = [super init];
    if(self) {
    count = 23;
    }
    return self;
    }
    - (int *) getCountPtr {
    return &count;
    }
    @end

上面的代码定义了一个 PointerBridge 类,它包含 getCountPtr 方法,这个方法返回一个值为 23 的 int 型内存地址。 这个 Int 其实是 count 的实例,它在构造方法 init 中被赋值为 23 。

我把这段代码放在一个 Objective-C 的头文件中,然后把这个头文件 import 到我的桥接头文件(XXX-bridging-header.h)中,这样就可以在 Swift 中使用。然后我在 Swift 中创建一个名为 bridge 的 PointerBridge 实例,然后获得 getCountPtr() 方法的返回值…

  1. 1
    2
    3
    4
  1. let bridge = PointerBridge()
    let theInt = bridge.getCountPtr()
    print(theInt)
    print(theInt.memory)

在 Xcode 中按住 Option 键点击 theInt 检查它的类型,你会发现他的 Swift 类型是 UnsafeMutablePointer<Int32>。这是指向 Int 型的指针,和 Int 型不一样,它仅仅是指向它的指针

如果运行这个程序然后执行这段 Swift 代码,我们会发现 theInt 在命令行中输出类似 0x00007f8bdb508ef8 这样的内存地址,然后,然后我们会看到 memory 成员变量输出的值 23 。访问指针指向的内存通常返回其底层指向的对象,在这个例子中就是原来的 32 位 int(在 Swift 中就是 Int32

现在让 Objective-C 类支持设置 count 的值。

  1. 1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
  1. @interface PointerBridge : NSObject {
    int count;
    }
    - (int *) getCountPtr;
    - (void) setCount:(int)newCount;
    @end
  2.  
  3. @implementation PointerBridge
    - (instancetype) init {
    self = [super init];
    if(self) {
    count = 23;
    }
    return self;
    }
    - (int *) getCountPtr {
    return &count;
    }
    - (void) setCount:(int)newCount {
    count = newCount;
    }
    @end

我们可以调用 setCount() 方法来修改 count 的值。因为 theInt 是一个指针,所以通过 setCount 修改 count 也会更新 theInt.memory。别忘了内存地址是不会变的,变的是值。

也就是说,下面的代码会在命令行中打印数字 23, 然后打印数字 1000。

  1. 1
    2
    3
    4
    5
  1. let bridge = PointerBridge()
    let theInt = bridge.getCountPtr()
    print(theInt.memory) // 23
    bridge.setCount(1000)
    print(theInt.memory) // 1000

如果想避免每次都写 .memory,有一条捷径就是把.memory 赋值给一个变量:

  1. 1
    2
    3
    4
  1. let bridge = PointerBridge()
    let theInt = bridge.getCountPtr()
    let countVal = theInt.memory
    print(countVal) // 23

就像之前一样, 命令行会输出 23。然而,如果我们像之前那样调用 setCount() 方法修改 count 的值,问题出现了:

  1. 1
    2
    3
    4
    5
    6
    7
  1. let bridge = PointerBridge()
    let theInt = bridge.getCountPtr()
    let countVal = theInt.memory
    print(countVal) // 23
  2.  
  3. bridge.setCount(1000)
    print(countVal) // 23

出现问题的原因是 countVal 是通过值(value)来赋值的。赋值的时候值(value)就是23,所以 countVal 有它自己的内存地址,这个地址永久地保存了 23 这个值,所以已经失去了指针的特性。 countVal 现在只是一个普通的 Int32 型。

在 Swift 中创建 C 指针

如果我们想要做和上面相反的事情呢?不是用 Int 型来给 count 赋值,而是传入一个指针呢?

我们假设在 Objective-C 的代码中有如下的一个方法:

  1. 1
    2
    3
  1. - (void) setCountPtr:(int *)newCountPtr {
    count = *newCountPtr;
    }

这个方法很作,其实就是把 newCountPtr 重新赋值给 count,但在 Swift 开发中你确实会碰到这样一些需要传入指针的场景。用这么作的方式只是为了向你展示如何在 Swift 中创建指针类型,然后传入到 Objective-C 的方法中。

你可能会简单的认为只要使用一个类似 & 的引用操作符就可以传入 Int 值,就像你在 C 中所做的那样。在 Objective-C 中你可以这样写:

  1. 1
    2
  1. int mcount = 500;
    [self setCountPtr:&mcount];

这段代码可以成功地把 count 的值更新为 500。然而在 Swift 中,通过自动补全你会发现它更复杂(而且更冗长)。它需要传入一个UnsafeMutablePointer<Int32> 类型的 newCountPtr 变量。

我知道这个类型很恶心,而且它看起来确实很复杂。但是,事实上它相当简单,特别是在你了解 Obj-C 中的指针的情况下。如果要创建一个UnsafeMutablePointer<Int32> 类型的对象,我们只需要调用构造方法,这个构造方法唯一需要传入的参数就是指针的大小(你应该知道 C 的指针是不存储类型的,所以它也不会存储大小的信息)

  1. 1
    2
    3
    4
    5
    6
    7
    8
    9
  1. let bridge = PointerBridge()
    let theInt = bridge.getCountPtr()
    print(theInt.memory) // 23
  2.  
  3. let newIntPtr = UnsafeMutablePointer<Int32>.alloc(1)
    newIntPtr.memory = 100
    bridge.setCountPtr(newIntPtr)
  4.  
  5. print(theInt.memory) // 100

唯一需要给 UnsafeMutablePointer<Int32> 构造方法传入的参数就是需要分配空间的对象的个数,所以我们传入 1 即可,因为我们只需要一个Int32 对象 。然后,只需要把我们之前对 memory 所做的事情反过来,我们就可以为我们新建的指针赋值。最终,我们只需要简单滴把 newIntPtr 传入到 setCountrPtr 方法中,再把之前 theInt 指针的值打印出来,我们就会发现它的值已经被更新为 100。

总结

UnsafeMutablePointer<T> 类型的兄弟类型 UnsafePointer<T> 从根本上说只是 C 指针的一个抽象。你可以把它们看作 Swift 的可选类型,这样更容易理解。它们不是直接等于一个确切的值,而是在一个确切的值上面做了一层抽象。它们的类型是泛型,这样就可以允许其使用其他的值,而不单单是 Int32。比如你需要传入一个 Float 对象那么你可能需要 UnsafeMutablePointer<Float>

重点是:你不是把一个 Int 强转 为 UnsafeMutablePointer<Int>,因为指针不是简单地一个 Int 值。所以,如果需要创建一个新的对象,你需要调用构造方法 UnsafeMutablePointer<Int>(count: Int)

在本文之后我们会继续深入研究函数指针的一些细节,然后学习如何利用这些特性的优势去更好地与 C 和 Objective-C 的 API 进行交互。一定要注册我们的 Newsletter ,这样你才不会错过这些精彩的内容!

浅谈 Swift 2 中的 Objective-C 指针的更多相关文章

  1. 转: 浅谈C/C++中的指针和数组(二)

    转自:http://www.cnblogs.com/dolphin0520/archive/2011/11/09/2242419.html 浅谈C/C++中的指针和数组(二) 前面已经讨论了指针和数组 ...

  2. 转:浅谈C/C++中的指针和数组(一)

    再次读的时候实践了一下代码,结果和原文不一致 error C2372: 'p' : redefinition; different types of indirection 不同类型的间接寻址 /// ...

  3. 转载 浅谈C/C++中的static和extern关键字

    浅谈C/C++中的static和extern关键字 2011-04-21 16:57 海子 博客园 字号:T | T   static是C++中常用的修饰符,它被用来控制变量的存贮方式和可见性.ext ...

  4. 浅谈C语言中的强符号、弱符号、强引用和弱引用

    摘自http://www.jb51.net/article/56924.htm 浅谈C语言中的强符号.弱符号.强引用和弱引用 投稿:hebedich 字体:[增加 减小] 类型:转载 时间:2014- ...

  5. 【sql注入】浅谈sql注入中的Post注入

    [sql注入]浅谈sql注入中的Post注入 本文来源:i春秋学院 00x01在许多交流群中,我看见很多朋友对于post注入很是迷茫,曾几何,我也是这样,因为我们都被复杂化了,想的太辅助了所以导致现在 ...

  6. 浅谈关于QT中Webkit内核浏览器

    关于QT中Webkit内核浏览器是本文要介绍的内容,主要是来学习QT中webkit中浏览器的使用.提起WebKit,大家自然而然地想到浏览器.作为浏览器内部的主要构件,WebKit的主要工作是渲染.给 ...

  7. 浅谈JAVA GUI中,AWT与Swing的区别、联系及优缺点

    浅谈JAVA GUI中,AWT与Swing的区别.联系及优缺点 A.区别 1.发布的时间 AWT是在JDK 1.0版本时提出的 Swing是在AWT之后提出的(JAVA 2) 2. ”重量” AWT是 ...

  8. c#Winform程序调用app.config文件配置数据库连接字符串 SQL Server文章目录 浅谈SQL Server中统计对于查询的影响 有关索引的DMV SQL Server中的执行引擎入门 【译】表变量和临时表的比较 对于表列数据类型选择的一点思考 SQL Server复制入门(一)----复制简介 操作系统中的进程与线程

    c#Winform程序调用app.config文件配置数据库连接字符串 你新建winform项目的时候,会有一个app.config的配置文件,写在里面的<connectionStrings n ...

  9. 浅谈人脸识别中的loss 损失函数

    浅谈人脸识别中的loss 损失函数 2019-04-17 17:57:33 liguiyuan112 阅读数 641更多 分类专栏: AI 人脸识别   版权声明:本文为博主原创文章,遵循CC 4.0 ...

随机推荐

  1. c# IOCP.ClientEx2.ReadWrite 加断点遭遇System.AccessViolationException 问题

    起因: 如果在Debug模式下,在IOCP.ClientEx2.ReadWrite.cs while (0 > (nPackSize = _ipcp.Pack(arg_n64PackId, ar ...

  2. IT兄弟连 JavaWeb教程 Servlet会话跟踪 Session常用方法

    ●  public Object getAttribute(String name) 该方法返回在该session会话中具有指定名称的对象,如果没有指定名称的对象,则返回null. ●  public ...

  3. tpc-ds99 工具使用

    安装部署 tpc-ds-99 工具 解压文件 unzip tpc-ds-tool.zip 进入目录 cd v2.3.0/tools 拷贝Makefile文件 cp Makefile.suite Mak ...

  4. stringstream转换

    在这之前,在杭电刷题的时候,并没有注意到这个好东西. 使用stringstream对象简化类型转换C++标准库中的<sstream>提供了比ANSI C的<stdio.h>更高 ...

  5. JTextArea设置滚动条

    应将JTextArea置于JScrollPanel中 若要使只有垂直滚动条而没有水平滚动条,使用JTextArea.setLineWrap(true),自动换行.   以下摘自[url]http:// ...

  6. IP服务-5-网络时间协议

    NTP版本3(RFC1305)允许IP主机向一个通用的时钟源同步它们的日期和时间. 从设计上来说,大多数路由器和交换机都使用NTP客户端模式,根据NTP服务器所提供的时间来调整自己的时钟.NTP定义了 ...

  7. 整理一下 通知传值 Block传值

    Block: 一. (1) 在需要传值的界面定义属性 // 点击collectionViewCell的回调 @property (nonatomic, copy) void(^Didcollectio ...

  8. for循环 与 for in 循环

    json是js里的一种数据格式.var obj={a:15,b:8,c:12}         这是个json对象 var arr=[15,8,12]; 数组alert(obj.a); ---15al ...

  9. Influxdb 时序数据库 windows 安装

    Influxdb 是一款比较火爆的时序数据库,本文介绍如何在 windows 平台下安装. 1.场景: windows 平台的 influxdb 似乎只支持单机非windows 服务的安装方式 适用于 ...

  10. vue初级学习--使用 vue-resource 请求数据

    一.导语 我发现好像我最近几次写文,都是在7号,很恰巧啊~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ...