原創文章,歡迎轉載。轉載請注明:關東升的博客
內存托管對象
Swift中調用Core Foundation函數獲得對象時候,對象分為:內存托管對象和內存非托管對象。
內存托管對象就是由編譯器幫助管理內存,我們不需要調用CFRetain函數獲得對象所有權,也不需要調用CFRelease函數放棄對象所有權。
獲得這些內存托管對象的方法,是采用了CF_RETURNS_RETAINED或CF_RETURNS_NOT_RETAINED注釋聲明,示例代碼:
-(CGPathRef)makeToPath CF_RETURNS_RETAINED
{
UIBezierPath* triangle = [UIBezierPath bezierPath];
[triangle moveToPoint:CGPointZero];
[triangle addLineToPoint: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);
}
內存托管對象使用起來比較簡單,不需要我們做額外的事情。
func CFStringCreateWithCString(_ alloc: CFAllocator!,
_ cStr: UnsafePointer<Int8>,
_ encoding: CFStringEncoding) -> CFString! //內存托管對象
func CFHostCreateCopy(_ alloc: CFAllocator?,
_ host: CFHost) -> Unmanaged<CFHost> //內存非托管對象
內存非托管對象
內存非托管對象就是內存需要程序員自己管理。這是由于在獲得對象的方法中沒有使用CF_RETURNS_RETAINED或CF_RETURNS_NOT_RETAINED注釋聲明,編譯器無法幫助管理內存。在具體使用時候我們可以上一節的方法判斷是否為非內存托管對象。
內存非托管對象使用起來有些麻煩,要根據獲得所有權方法,進行相應的處理。
如果一個函數名中包含Create或Copy,則調用者獲得這個對象的同時也獲得對象所有權,返回值Unmanaged<T>需要調用takeRetainedValue()方法獲得對象。調用者不再使用對象時候,Swift代碼中需要調用CFRelease函數放棄對象所有權,這是因為Swift是ARC內存管理的。
如果一個函數名中包含Get,則調用者獲得這個對象的同時不會獲得對象所有權,返回值Unmanaged<T>需要調用takeUnretainedValue()方法獲得對象。
示例代碼如下:
let host: CFHost = CFHostCreateWithName(kCFAllocatorDefault,
?"127.0.0.1").takeRetainedValue()
let hostNames: CFArray = CFHostGetNames(host, nil)!.takeUnretainedValue()