内存分区:(堆区,栈区,静态区,常量区,代码区)
1、栈区 (heap)
- 存放的局部变量、先进后出、一旦出了作用域就会被销毁;函数跳转地址,现场保护等;
- 程序猿不需要管理栈区变量的内存;
-栈区地址从高到低分配;
2、堆区(stack)
堆区的内存分配使用的是alloc;
需要程序猿管理内存;
ARC的内存的管理,是编译器再便宜的时候自动添加 retain、release、autorelease;
堆区的地址是从低到高分配)
3、静态区/全局区(static)
包括两个部分:未初始化过 、初始化过;
也就是说,(全局区/静态区)在内存中是放在一起的,初始化的全局变量和静态变量在一块区域, 未初始化的全局变量和未初始化的静态变量在相邻的另一块区域;
eg:int a;未初始化的。int a = 10;已初始化的。
4、常量区:
常量字符串就是放在这里
5、代码区:
存放App代码:
在iOS中,堆区的内存是应用程序共享的,堆中的内存分配是系统负责的;
系统使用一个链表来维护所有已经分配的内存空间(系统仅仅纪录,并不管理具体的内容);
变量使用结束后,需要释放内存,OC中是根据引用计数==0,就说明没有任何变量使用该空间,那么系统将直接收回;
当一个app启动后,代码区,常量区,全局区大小已固定,因此指向这些区的指针不会产生崩溃性的错误。而堆区和栈区是时时刻刻变化的(堆的创建销毁,栈的弹入弹出),所以当使用一个指针指向这两个区里面的内存时,一定要注意内存是否已经被释放,否则会产生程序崩溃(也即是野指针报错)。
其它操作系统:
iOS是基于UNIX、Android是基于Linux的,在Linux和unix系统中,内存管理的方式基本相同;
Android应用程序的内存分配也是如此。除此以外,这些应用层的程序使用的都是虚拟内存,它们都是建立在操作系统之上的,只有开发底层驱动或板级支持包时才会接触到物理内存;
举例:在嵌入式Linux中,实际的物理地址只有64M甚至更小,但是虚拟内存却可以高达4G;