阅读更多

0顶
0踩

移动开发

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

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

  • 关于组织参加“第八届‘泰迪杯’数据挖掘挑战赛”的通知-4页

    关于组织参加“第八届‘泰迪杯’数据挖掘挑战赛”的通知-4页

  • PyMySQL-1.1.0rc1.tar.gz

    PyMySQL-1.1.0rc1.tar.gz

  • 技术资料分享CC2530中文数据手册完全版非常好的技术资料.zip

    技术资料分享CC2530中文数据手册完全版非常好的技术资料.zip

  • docker构建php开发环境

    docker构建php开发环境

  • VB程序实例59_系统信息_显示分辨率.zip

    VB程序实例,可供参考学习使用,希望对你有所帮助

  • pytz-2016.7-py2.6.egg

    pytz库的主要功能 时区转换:pytz库允许用户将时间从一个时区转换到另一个时区,这对于处理跨国业务或需要处理多地时间的数据分析尤为重要。 历史时区数据支持:pytz库不仅提供了当前的时区数据,还包含了历史上不同时期的时区信息,这使得它在处理历史数据时具有无与伦比的优势。 夏令时处理:pytz库能够自动处理夏令时的变化,当获取某个时区的时间时,它会自动考虑是否处于夏令时期间。 与datetime模块集成:pytz库可以与Python标准库中的datetime模块一起使用,以确保在涉及不同时区的场景中时间的准确性。

  • VB程序实例-为程序添加快捷键.zip

    VB程序实例-为程序添加快捷键.zip

  • 画2、3维的隐含数111111111111

    画2、3维的隐含数

  • pytz-2017.2-py3.4.egg

    pytz库的主要功能 时区转换:pytz库允许用户将时间从一个时区转换到另一个时区,这对于处理跨国业务或需要处理多地时间的数据分析尤为重要。 历史时区数据支持:pytz库不仅提供了当前的时区数据,还包含了历史上不同时期的时区信息,这使得它在处理历史数据时具有无与伦比的优势。 夏令时处理:pytz库能够自动处理夏令时的变化,当获取某个时区的时间时,它会自动考虑是否处于夏令时期间。 与datetime模块集成:pytz库可以与Python标准库中的datetime模块一起使用,以确保在涉及不同时区的场景中时间的准确性。

Global site tag (gtag.js) - Google Analytics