阅读更多

0顶
0踩

移动开发

转载新闻 浅谈Swift 2中的Objective-C指针

2015-09-08 11:20 by 副主编 mengyidan1988 评论(0) 有5577人浏览
本文出自:JamesonQuave.com,作者:Jameson Quave,译文出自:SwiftGG,译者:mmoaay
在Objective-C中我们经常会用到指针,有些方法也需要直接去操作指针,今天我们就来看看如何在Swift中使用指针。

在Swift中读C指针

下面这个Objective-C方法会返回一个int指针,或者说C术语里面的(int *):
@interface PointerBridge : NSObject {
    int count;
}
- (int *) getCountPtr;
@end
 
@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() 方法的返回值…
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的值。
@interface PointerBridge : NSObject {
    int count;
}
- (int *) getCountPtr;
- (void) setCount:(int)newCount;
@end
 
@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。
let bridge = PointerBridge()
let theInt = bridge.getCountPtr()
print(theInt.memory) // 23
bridge.setCount(1000)
print(theInt.memory) // 1000

如果想避免每次都写.memory,有一条捷径就是把.memory赋值给一个变量:
let bridge = PointerBridge()
let theInt = bridge.getCountPtr()
let countVal = theInt.memory
print(countVal) // 23

就像之前一样, 命令行会输出23。然而,如果我们像之前那样调用setCount()方法修改count的值,问题出现了:
let bridge = PointerBridge()
let theInt = bridge.getCountPtr()
let countVal = theInt.memory
print(countVal) // 23
 
bridge.setCount(1000)
print(countVal) // 23

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

在Swift中创建C指针

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

我们假设在Objective-C的代码中有如下的一个方法:
- (void) setCountPtr:(int *)newCountPtr {
    count = *newCountPtr;
}

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

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

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

我知道这个类型很恶心,而且它看起来确实很复杂。但是,事实上它相当简单,特别是在你了解Objective-C中的指针的情况下。如果要创建一个UnsafeMutablePointer<Int32> 类型的对象,我们只需要调用构造方法,这个构造方法唯一需要传入的参数就是指针的大小(你应该知道C的指针是不存储类型的,所以它也不会存储大小的信息)。
let bridge = PointerBridge() let theInt = bridge.getCountPtr() print(theInt.memory)
        // 23 let newIntPtr = UnsafeMutablePointer<Int32>.alloc(1) newIntPtr.memory
        = 100 bridge.setCountPtr(newIntPtr) 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)。
0
0
评论 共 0 条 请登录后发表评论

发表评论

您还没有登录,请您登录后再发表评论

相关推荐

  • 浅谈 Swift 2 中的 Objective-C 指针

    浅谈 Swift 2 中的 Objective-C 指针 2015-09-07499 文章目录 1.在 Swift 中读 C 指针 2.在 Swift 中创建 C 指针 3.总结 作者:Jameson Quave,原文链接,原文日期:2015/08/23译者:mmoaay;校对:...

  • 浅谈Swift和Objective-C之间的那点事。。。

    Swift,苹果于2014年WWDC(苹果开发者大会)发布的新开发语言,可与Objective-C*共同运行于Mac OS和iOS平台,用于搭建基于苹果平台的应用程序。

  • 来一次有侧重点的区分Swift与Objective-C

    面试中经常被问到Objective-C与Swift的区别,其实区别还是很多的,重点整理一下个人觉得很重要的:面向协议编程。 一、Objective-C与Swift的异同 1.1、swift和OC的共同点: – OC出现过的绝大多数概念,比如引用...

  • 浅谈Swift语法

    好了,言归正传,这里,我从一个iOS开发人员的角度,来浅谈一下Swift的语法! 首先,一个很简单的用法,输出函数: println("Hello, world") 从这段代码里,我们可以发现这个函数名好像在Java里见...

  • Objective-C 内存管理

    这么写是有问题的,这种是浅拷贝,只是把指针地址赋值过去了,它的引用计数并没有变化,这样两个指针指向了一块引用计数为1的内存空间,有引起野指针的潜在风险 ,那么为了避免这个问题,我们给newValue调用...

  • 浅谈Swift和OC的区别

    前言 一晃Swift3都出来快一年了,从OC到Swift也经历了很多,所以对两者的一些使用区别也总结了一点,暂且记录下,权当自己的一个笔记。 当然其中一些区别可能大家都有耳闻,所以...Swift和Objective-C共用一套运行...

  • 浅谈iOS性能优化之APP崩溃与hook方案

    如何收集crash 利用bugly、友盟等第三方收集 监控crash原理 防崩溃处理 常见崩溃类型 防崩溃处理方案 hook方案 安全接口 如何收集crash 在平常开发过程中,由于代码的不严谨比如不对入参做校验,使用C++野指针等会...

  • Objective-C内存管理:Block

    总述 以下环境都在ARC环境下,常规设置,使用...Block中__block关键字为何能同步Block外部和内部的值? Block有几种类型? 什么时候栈上的Block会复制到堆? Block的循环引用应该如何处理? Block外部__we...

  • 浅谈iOS KVO

    KVO 是基于Runtime机制实现的,KVO是运用了一个isa-swizzling 技术,就是类型混合指针机制, 将2个对象的isa指针互相调换, 就是俗称的黑魔法. 当某个类的属性对象第一次被观察时,系统就会在运行期动态地创建该类的...

  • 浅谈苹果新开发语言:Swift

     今天是个值得纪念的日子:因为苹果的WWDC大会。苹果的每次WWDC(全球开发者大会)举行都让我们像打了肾上腺素这么兴奋、幸福、惊叹、震撼、深思。...为了不至于太落后受鄙视,我今天也来简单谈

  • Swift 浅谈Struct与Class

    讨论Struct与Class之前,我们先来看一个概念:Value Type...2. 引用类型的变量存储对他们的数据引用,因此后者称为对象,因此对一个变量操作可能影响另一个变量所引用的对象。 这就是我们之前博客中提到的深拷贝...

  • 下载的资源,下载后解压

    python->exe

  • 前段web开发实战-哪吒2哪吒闹海网站(超酷哪吒主题网页,HTML+CSS 带你畅游)

    宝子们,快来看看这个超炫的哪吒主题网页!用 HTML 和 CSS 精心打造,有酷炫的头部导航、精彩的横幅内容,还有各种哪吒相关人物介绍和超燃票房海报。代码简单易懂,非常适合前端小白学习,一起动手搭建属于自己的哪吒世界吧!

  • Java毕业设计-SpringBoot+Vue的家乡特色推荐系统(附源码、数据库、教程).zip

    Java 项目, Java 毕业设计,Java 课程设计,基于 SpringBoot 开发的,含有代码注释,新手也可看懂。毕业设计、期末大作业、课程设计、高分必看,下载下来,简单部署,就可以使用。 包含:项目源码、数据库脚本、软件工具等,前后端代码都在里面。 该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值。 项目都经过严格调试,确保可以运行! 1. 技术组成 前端:html、javascript、Vue 后台框架:SpringBoot 开发环境:idea 数据库:MySql(建议用 5.7 版本,8.0 有时候会有坑) 数据库工具:navicat 部署环境:Tomcat(建议用 7.x 或者 8.x 版本), maven 2. 部署 如果部署有疑问的话,可以找我咨询 Java工具包下载地址: https://pan.quark.cn/s/eb24351ebac4 后台路径地址:localhost:8080/项目名称/admin/dist/index.html 前台路径地址:localhost:8080/项目名称/front/index.html (无前台不需要输入)

  • 2023年专升本计算机复习题.pdf

    2023年专升本计算机复习题.pdf

  • 基于SSM+JSP的班级同学录网站+数据库(Java毕业设计,包括源码,教程).zip

    Java 项目, Java 毕业设计,Java 课程设计,基于 SpringBoot 开发的,含有代码注释,新手也可看懂。毕业设计、期末大作业、课程设计、高分必看,下载下来,简单部署,就可以使用。 包含:项目源码、数据库脚本、软件工具等,前后端代码都在里面。 该系统功能完善、界面美观、操作简单、功能齐全、管理便捷,具有很高的实际应用价值。 项目都经过严格调试,确保可以运行! 1. 技术组成 前端:jsp 后台框架:SSM 开发环境:idea 数据库:MySql(建议用 5.7 版本,8.0 有时候会有坑) 数据库工具:navicat 部署环境:Tomcat(建议用 7.x 或者 8.x 版本), maven 2. 部署 如果部署有疑问的话,可以找我咨询 Java工具包下载地址: https://pan.quark.cn/s/eb24351ebac4

  • 【工程项目】MATLAB虫害监测(带识别率).zip

    【工程项目】MATLAB虫害监测(带识别率)

  • 金刚石化学机械抛光(CMP)中层状双氢氧化物催化作用研究及应用

    内容概要:本文详细研究了金刚石(Single Crystal Diamond, SCD)化学机械抛光(CMP)过程中,抛光浆料的组成成分对其表面质量的影响。通过对浆料的化学成分(氧化剂种类及浓度、催化剂种类)及机械因素(磨料粒径)进行系统性探索,研究揭示了氧化剂、磨料和催化剂之间复杂的相互作用关系。重点介绍了层状双氢氧化物(Layered Double Hydroxides, LDH)的应用和芬顿反应机理,探讨其对提高抛光效果的独特贡献,特别是表面粗糙度和质量的提升。实验中通过一系列对照测试比较不同浆料配置对抛光结果的不同影响,得出了具体的最优配置参数,实现了Ra约为0.109 nm的原子级平整表面。 适合人群:从事CMP工艺研究的专业人士、材料科学及物理学相关背景研究人员和技术人员、关注高端材料制备与处理的企业研发人员。 使用场景及目标:适用于需要深入了解CMP技术细节及应用的科学研究项目;针对希望提升超硬材料如金刚石表面加工品质的实际工业生产环境。本研究所建立的最佳浆料配比及相应抛光参数对实现高效能CMP处理流程有着重要的指导意义,尤其有助于优化半导体器件制造和其他高精度要求的技术

  • 基于JAVA的机场航班起降与协调管理系统&毕业设计&毕业论文&数据库&演示视频&源代码

    本次项目是设计一个基于JAVA的机场航班起降与协调管理系统。 (1)在经济可行性上来分析的话,该软件是机场内部使用的一个指挥协调软件,属于航空安全投资,本软件开发成本并不高,软件和服务器数据库可以用机场原有的数据库进行开发,比起空难给航空公司造成的损失来说九牛一毛。 (2)在技术可行性上来分析的话,该软件主要运用了Java技术、jQuery-easyui和Mysql数据库技术。Java是到目前来说最稳定的、最可靠的软件开发工具;jQuery-easyui虽然是比较新的前台开发技术,但是他的界面新颖整洁,适合于功能性软件的开发;Mysql数据库也是许多大公司都采用的软件项目开发数据库,不仅稳定而且性能可靠,可以用作本次软件的开发。 (3)在法律可行性上来分析的话,该软件使用的技术都为开源的软件开发工具和语言,虽然Java等开发技术都存在Sun公司的版权问题,但是Java技术是可以免费使用的,没有涉及到法律上的侵权。 (4)在方案可行性上来分析的话,此次软件开发的很大一部分精力都放在了软件的需求分析和设计方面,设计出来的软件可以很好地去实现我们所要完成的软件预先设计的功能。

  • 2023年计算机组成与系统结构实验报告.pdf

    2023年计算机组成与系统结构实验报告.pdf

Global site tag (gtag.js) - Google Analytics