内存管理(上)

内存管理

  • ARC是Swift默认的内存管理机制,其针对堆上的对象,由编译器自动生成操作引用计数的指令(retain或release)来管理对象的引用计数增加或减少。程序员无法手工控制。
  • 哪些类型的对象收ARC管理
    • 类class
    • 闭包closure
  • 哪些对象不受ARC管理
    • 基础数值类型
    • 结构struct
    • 枚举enum
    • 元组tuple
class FileStream{
  init(){
    print("open file...")
  }    
  func process(){
    print("process file...")
  }
  deinit{
    print("close file...")
  }
}
class MyClass1{
  var stream:FileStream?
}
class MyClass2{
  static var stream:FileStream?
}
var fs1:FileStream() // r(1)

var mc:MyClass1?
mc!.stream=fs1 // r(2)

MyClass2.stream=fs1 // r(3)

fs1?.process()

fs1=nil //r(2)
mc!.stream=nil //r(1)
Myclass2.stream=nil // r(0)

图示引用计数.png

关于Auto Release

  • Swift原生对象不支持autorelease消息,也就没有必要使用Autorelease Pool来管理内存峰值。
  • 如果Swift调用Objective-C函数返回autorelease对象,那么如果出现内存峰值过高的风险,仍然需要使用Autorelease Pool来管理。
  • Autorelease Pool在Swift中使用尾随闭包形式实现。
import Foundation

class FileStream{
  init(){
    print("open file...")
  }    
  func process(){
    print("process file...")
  }
  deinit{
    print("close file...")
  }
}
func getStream()-> FileStream?{
  var fs:FileStream?

  fs=FileStream()
  return fs;
}
func useNativeSwiftObject(){
  for index in 0...10{
    print("\(index) for start...")
      var stream=getStream()
      stream?.process()
      stream=nil
      print("\(index) for end...\n")
  }
}
/*
func useObjecAutoreleaseObject(){
  if let filepath = NSBundle.mainBundle().pathForResource("2015",ofType:"jpg"){
    for _ in 0..<5{
      autoreleasepool{
        for _ in 0..<1000{
          let data=NSdata.dataWithContentsOfMappedFile(filepath)
        }
      }
    }
  }
}
*/
print("\n......invoke......\n")
useNativeSwiftObject()
print("......continue......")

循环引用于内存泄露

  • 对象间的循环引用会造成ARC引用计数无法释放被引用的任何一个对象,从而造成内存泄露。

  • 解决循环引用造成的内存泄露有三种方式:
    1.在合适的地方,手动将循环引用解除。
    2.如果允许对象引用为nil,可将引用声明为弱引用(weak)
    3.如果不允许对象引用为nil,可将对象声明为无主引用(unowned)

  • 弱引用在对象被释放后,ARC会将引用设置为nil。无主引用在对象被释放后,ARC不会设置nil,访问时会抛运行时错误(空悬指针)。

class Computer{
  var name: String
  var display: Monitor?
  init(name:String){
    self.name=name
    print("Computer init")
  }
  deinit{
    print("Computer deinit")
  }
}
class Monitor{
  var no: Int
  weak var device: Computer?
  init(no:Int){
    self.no=no
    print("Monitor init")
  } 
  deinit{
    print("Monitor deinit")
  }
}

var imac:Computer?
var screen:Monitor?
imac=Computer(name:"Jason's iMac")
screen=Monitor(no:29)

imac!.display=screen
screen!.device=imac

//imac!.display=nil

imac=nil
screen=nil

循环引用.png
避免循环引用:弱引用.png
import Foundation

class Car{
  var name:String
  var engine:Engine?
  init(name:String){
    self.name=name
    print("Car init")
  }
  deinit{
    print("Car deinit")
  }
}
class Engine{
  var no:Int
  unowned var car : Car
  init(no:Int,car : Car){
    self.no=no
    self.car=car
    print("Engine init")
  }
  deinit{
    print("Engine deinit")
  }
}
var bmw:Car?
var engine:Engine?

bmw=Car(name:"BMW")
engine=Engine(no:400,car:bmw!)

bmw!.engine=engine

bmw=nil
//engine=nil
避免循环引用:无主引用.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 内存管理 简述OC中内存管理机制。与retain配对使用的方法是dealloc还是release,为什么?需要与a...
    丶逐渐阅读 2,029评论 1 16
  • 29.理解引用计数 Objective-C语言使用引用计数来管理内存,也就是说,每个对象都有个可以递增或递减的计数...
    Code_Ninja阅读 1,558评论 1 3
  • 一.开篇之初 内存管理解决的问题就是:1)防止野指针的生成(野指针:指向变量的指针还存在,但是所指向的内存已经被释...
    半纸渊阅读 1,039评论 0 1
  • iOS开发中, 之前一直使用swift, 因此对于Objective-C的内存管理机制长期处于混乱的一知半解状态....
    icetime17阅读 886评论 1 8
  • 为什么进行内存管理? 由于移动设备的内存极其有限,所以每个APP所占的内存也是有限制的,当app所占用的内存较多时...
    天天想念阅读 920评论 1 7