《从零开始学Swift》学习笔记(Day 62)——Core Foundation框架之内存托管对象与非托管对象
程序员文章站
2022-05-09 20:32:26
Swift中调用CoreFoundation函数获得对象时候,对象分为:内存托管对象和内存非托管对象。
内存托管对象就是由编译器帮助管理内存,我们不需要调用CFRetai...
Swift中调用CoreFoundation函数获得对象时候,对象分为:内存托管对象和内存非托管对象。
内存托管对象就是由编译器帮助管理内存,我们不需要调用CFRetain函数获得对象所有权,也不需要调用CFRelease函数放弃对象所有权。
获得这些内存托管对象的方法,是采用了CF_RETURNS_RETAINED或CF_RETURNS_NOT_RETAINED注释声明,示例代码:
-(CGPathRef)makeToPathCF_RETURNS_RETAINED { UIBezierPath* triangle = [UIBezierPathbezierPath]; [triangle moveToPoint:CGPointZero]; [triangleaddLineToPoint:CGPointMake(self.view.frame.size.width,0)]; [triangle addLineToPoint:CGPointMake(0,self.view.frame.size.height)]; [triangle closePath]; CGPathRef theCGPath = [triangle CGPath]; return CGPathCreateCopy(theCGPath); }
内存托管对象使用起来比较简单,不需要我们做额外的事情。
funcCFStringCreateWithCString(_ alloc: CFAllocator!, _ cStr: UnsafePointer<Int8>, _encoding: CFStringEncoding) -> CFString! //内存托管对象 func CFHostCreateCopy(_alloc: CFAllocator?, _host: CFHost) -> Unmanaged<CFHost> //内存非托管对象
内存非托管对象
内存非托管对象就是内存需要程序员自己管理。这是由于在获得对象的方法中没有使用CF_RETURNS_RETAINED或CF_RETURNS_NOT_RETAINED注释声明,编译器无法帮助管理内存。在具体使用时候我们可以上一节的方法判断是否为非内存托管对象。
内存非托管对象使用起来有些麻烦,要根据获得所有权方法,进行相应的处理。
1. 如果一个函数名中包含Create或Copy,则调用者获得这个对象的同时也获得对象所有权,返回值Unmanaged需要调用takeRetainedValue()方法获得对象。调用者不再使用对象时候,Swift代码中需要调用CFRelease函数放弃对象所有权,这是因为Swift是ARC内存管理的。
2. 如果一个函数名中包含Get,则调用者获得这个对象的同时不会获得对象所有权,返回值Unmanaged需要调用takeUnretainedValue()方法获得对象。
示例代码如下:
let host: CFHost =CFHostCreateWithName(kCFAllocatorDefault, "127.0.0.1").takeRetainedValue() let hostNames: CFArray =CFHostGetNames(host, nil)!.takeUnretainedValue()