阅读更多

0顶
0踩

移动开发

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

2015-09-08 11:20 by 副主编 mengyidan1988 评论(0) 有5514人浏览
本文出自: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. 引用类型的变量存储对他们的数据引用,因此后者称为对象,因此对一个变量操作可能影响另一个变量所引用的对象。 这就是我们之前博客中提到的深拷贝...

  • 三菱FX3G FX3S与四台E700变频器Modbus RTU通讯控制:正反转、频率设定与读取方案,三菱FX3G FX3S与四台E700变频器通讯:Modbus RTU协议实现正反转、频率设定与控制

    三菱FX3G FX3S与四台E700变频器Modbus RTU通讯控制:正反转、频率设定与读取方案,三菱FX3G FX3S与四台E700变频器通讯:Modbus RTU协议实现正反转、频率设定与控制,快速反馈与教程包含,三菱FX3G FX3S 485协议通讯四台三菱E700变频器程序资料 三菱FX3G FX3S+485bd扩展,采用modbus rtu协议,crc校验,通讯控制四台E700变频器,可以实现正反转,停止,频率的设定,频率,电流等的读取。 反馈快,使用方便,包括教程,plc和触摸屏程序,变频器参数设置和接线,别的变频器支持rtu协议也可以实现。 ,三菱FX系列PLC; 485协议通讯; 变频器E700; 通讯控制; 参数设置; 教程。,三菱PLC控制E700变频器:485协议通讯与程序设置全解

  • hyphen-nl-0.20050617-10.el7.x64-86.rpm.tar.gz

    1、文件内容:hyphen-nl-0.20050617-10.el7.rpm以及相关依赖 2、文件形式:tar.gz压缩包 3、安装指令: #Step1、解压 tar -zxvf /mnt/data/output/hyphen-nl-0.20050617-10.el7.tar.gz #Step2、进入解压后的目录,执行安装 sudo rpm -ivh *.rpm 4、更多资源/技术支持:公众号禅静编程坊

  • 西门子S7-1200PLC结构化编程在5轴伺服项目中的应用:模块化设计、触摸屏控制及电气图纸实战解析,西门子S7-1200PLC结构化编程实现多轴联动与多种伺服功能应用:CAD图纸、PLC程序和触摸屏

    西门子S7-1200PLC结构化编程在5轴伺服项目中的应用:模块化设计、触摸屏控制及电气图纸实战解析,西门子S7-1200PLC结构化编程实现多轴联动与多种伺服功能应用:CAD图纸、PLC程序和触摸屏程序协同运作。,西门子S7-1200PLC结构化编程5轴伺服项目 ,包含plc程序、威纶通触摸屏程序、cad电气图纸。 可以实现以下功能,规格有: 1.三轴机械手X轴-Y轴-Z轴联动取放料PTO脉冲定位控制台达B2伺服 2.台达伺服速度模式应用+扭矩模式应用实现收放卷 3.程序为结构化编程,每一功能为模块化设计,功能:自动_手动_单步_暂停后原位置继续运行_轴断电保持_报警功能_气缸运行及报警. 4.每个功能块可以无数次重复调用,可以建成库,用时调出即可 5.上位机采样威纶通触摸屏 6.参考本案例熟悉掌握结构化编程技巧,扩展逻辑思维。 博图14以上都可以打开 ,核心关键词:西门子S7-1200PLC; 结构化编程; 5轴伺服项目; PLC程序; 威纶通触摸屏程序; CAD电气图纸; 三轴机械手; PTO脉冲定位控制; 台达B2伺服; 速度模式应用; 扭矩模式应用; 模块化设计; 轴断电保

  • 情感分析算法的关键应用领域与典型实战案例

    情感分析算法在多个领域有着广泛的应用场景和丰富的案例

  • 基于MATLAB仿真的MMC整流站与逆变站柔性互联技术研究:快速工况仿真与环流抑制控制,基于MATLAB仿真的MMC整流站与逆变站运行分析及四端柔性互联工况仿真模拟研究,21电平MMC整流站、MMC逆

    基于MATLAB仿真的MMC整流站与逆变站柔性互联技术研究:快速工况仿真与环流抑制控制,基于MATLAB仿真的MMC整流站与逆变站运行分析及四端柔性互联工况仿真模拟研究,21电平MMC整流站、MMC逆变站、两端柔性互联的MATLAB仿真模型,4端柔性互联、MMC桥臂平均值模型、MMC聚合模型(四端21电平一分钟即能完成2s的工况仿真) 1-全部能正常运行,图四和图五为仿真波形 2-双闭环控制,逆变站PQ控制,整流站站Udc Q控制 3-最近电平逼近调制+子模块电容充电 4-环流抑制控制 ,1. 21电平MMC整流站; 2. MMC逆变站; 3. MATLAB仿真模型; 4. 两端柔性互联; 5. 桥臂平均值模型; 6. 聚合模型; 7. 双闭环控制; 8. 最近电平逼近调制; 9. 子模块电容充电; 10. 环流抑制控制。,基于柔性互联的MMC系统仿真模型:多电平控制与环流抑制研究

  • 有效应对网络舆情教育培训PPT.pptx

    有效应对网络舆情教育培训PPT.pptx

  • 高光谱解混和图片去噪 附Matlab代码.rar

    1.版本:matlab2014/2019a/2024a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。

  • 【轴承压力】基于matlab GUI止推轴承压力计算【含Matlab源码 12069期】.zip

    Matlab领域上传的视频是由对应的完整代码运行得来的,完整代码皆可运行,亲测可用,适合小白; 1、从视频里可见完整代码的内容 主函数:main.m; 调用函数:其他m文件;无需运行 运行结果效果图; 2、代码运行版本 Matlab 2019b;若运行有误,根据提示修改;若不会,私信博主; 3、运行操作步骤 步骤一:将所有文件放到Matlab的当前文件夹中; 步骤二:双击打开main.m文件; 步骤三:点击运行,等程序运行完得到结果; 4、仿真咨询 如需其他服务,可私信博主; 4.1 博客或资源的完整代码提供 4.2 期刊或参考文献复现 4.3 Matlab程序定制 4.4 科研合作

  • 娱乐小工具微信小程序源码下载支持多种流量主.zip

    淘宝买的,直接分享给大家了,没有测试环境,也没有办法去测。但我想,他应该是可以用的

Global site tag (gtag.js) - Google Analytics