一般程序的内存分配,从高位到低位依次为

全局静态区:用于存储全局变量、静态变量等;这部分内存在程序编译时已经分配好,由操作系统管理,速度快,不易出错。

栈:函数中的基础类型的局部变量;由程序进行系统调用向操作系统申请,由操作系统管理,速度快。每个线程有自己的栈区。

堆:使用malloc或new申请的内存;由程序运行过程中动态分配任意大小的内存,由程序管理,使用free或者delete删除;频繁的分配和释放必然导致内存碎片。

常量区:存放常量字符串,程序结束后由系统释放。

程序代码区:存放程序的二进制代码。

Go的内存分配:

Go是内置运行时runtime的语言;像这种内置运行时的语言会抛弃传统的内存管理方式,改为自己管理;这样可以完成类似预分配,内存池等操作,以避开系统调用产生的性能问题。Go的内存分配可以分为以下几点:

1. 每次从操作系统申请一大块内存,由Go来对这块内存做分配,减少系统调用。

2. 内存分配算法采用 TCMalloc算法,核心思想是把内存分的非常细,进行分级管理,以降低锁的粒度。

3. 回收对象内存时,并没有将其真正释放掉,而是放回预先分配的大内存中,以便复用。只有内存闲置过多的时候,才会尝试归还部分内存给操作系统,降低整体开销。

Go在程序启动的时候,会分配一块连续的内存(虚拟内存),整体如下:

arena就是所谓的堆区,他把内存分割成 8K大小的页,页是heap中的最小管理单位,一些页组合起来形成了mspan。

bitmap区用于保存arena对应地址(指针大小为 8B,bitmap中一个byte大小的内存对应arena区域中的4个指针,因此大小为 512G/(4 * 8B))中是否保存了对象,以及对象是否被gc过,主要用于gc。

spans区域存放了mspan的指针,用于表示arena区的某一页属于哪个mspan。大小为 512G / 8KB(页的大小) * 8B(指针大小)。在创建 mspan的时候,按页填充对应的spans区域,在回收object时,很容易找到他所属的mspan。

Go内存管理结构

1. class和mspan

go对象管理粒度分为67种大小,也叫class和块。一个page根据class大小分为8K/class size。

class相同且地址连续的pages组成一个span。mspan是span的实际结构,mspan之间组成双向链表关联。

2. mcache 

每个工作线程都有各自的mcache,本地缓存可用的mspan资源,这样各个线程在申请内存的时候无需加锁,对于小于32KB大小的对象,直接通过mcache分配;对于大于23KB的大对象,则需要在mheap中分配。

每个goroutine对于同一个class都有scan和noscan两种mspan队列,其中scan分配给含有指针的对象,noscan分配给不含指针的对象,这样做是为了便于进行垃圾回收,在进行垃圾回收的时候,对于不包含指针的对象,无需进一步扫描是否引用其他活跃对象。

3. mcenter

mcenter为所有mcache提供切分好的mspan。有多少种mspan类型的mspan就有多少个mcenter,每个mcenter保存一种特定类型的mspan,提供两个mspan双端队列,包括已分配出去的和未分配出去的。

所有线程共享,需要加锁访问,但是申请一种 类型的mspan不会影响其他的mcenter.

type mcentral struct {
lock mutex
spanclass spanClass
nonempty mSpanList // list of spans with a free object, ie a nonempty free list
empty mSpanList // list of spans with no free objects (or cached in an mcache) // nmalloc is the cumulative count of objects allocated from
// this mcentral, assuming all spans in mcaches are
// fully-allocated. Written atomically, read under STW.
nmalloc uint64
}

4. mheap

代表Go程序的堆空间,Go程序使用一个mheap来管理堆空间。

当mcenter中没有空闲的mspan时,会向mheap申请。而mheap没有剩余内存时,会向操作系统申请新内存。

另外,对于大于32Kb的大对象,需要在mheap中分配,所有线程共享,需要加锁。

参考:

https://zhuanlan.zhihu.com/p/59125443

https://www.cnblogs.com/unqiang/p/12052308.html

https://zhuanlan.zhihu.com/p/128462868

go - 内存分配机制详解的更多相关文章

  1. ARC内存管理机制详解

    ARC在OC里面个人感觉又是一个高大上的牛词,在前面Objective-C中的内存管理部分提到了ARC内存管理机制,ARC是Automatic Reference Counting---自动引用计数. ...

  2. C++内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区

    栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清除的变量的存储区.里面的变量通常是局部变量.函数参数等.在一个进程中,位于用户虚拟地址空间顶部的是用户栈,编译器用它来实现函数的调用.和堆一样 ...

  3. 【校招面试 之 C/C++】第14题 C++ 内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区(堆栈的区别)

    栈,就是那些由编译器在需要的时候分配,在不需要的时候自动清除的变量的存储区.里面的变量通常是局部变量.函数参数等.在一个进程中,位于用户虚拟地址空间顶部的是用户栈,编译器用它来实现函数的调用.和堆一样 ...

  4. (转)C++内存分配方式详解——堆、栈、自由存储区、全局/静态存储区和常量存储区

    程序在内存有五个存在区域: A:动态区域中的栈区  B:动态区域中的栈区 C:静态区域中:全局变量 和静态变量    (这个区域又可以进一步细分为:初始化的全局变量和静态变量    以及    未初始 ...

  5. Java的内存回收机制详解X

    http://blog.csdn.net/yqlakers/article/details/70138786 1 垃圾回收的意义 在C++中,对象所占的内存在程序结束运行之前一直被占用,在明确释放之前 ...

  6. TLFS 内存分配算法详解

    文章目录 1. DSA 背景介绍 1.1 mmheap 1.2 mmblk 2. TLFS 原理 2.1 存储结构 2.2 内存池初始化 2.3 free 2.4 malloc 参考资料 1. DSA ...

  7. object-c(oc)内存管理机制详解

    1.内存的创建和释放 让我们以Object-c世界中最最简单的申请内存方式展开,谈谈关于一个对象的生命周期.首先创建一个对象: 1 2 3 //“ClassName”是任何你想写的类名,比如NSStr ...

  8. linux的vm.overcommit_memory的内存分配参数详解

    公司的redis有时background save db不成功,通过log发现下面的告警,很可能由它引起的: [13223] 17 Mar 13:18:02.207 # WARNING overcom ...

  9. Redis 内存淘汰机制详解

    一般来说,缓存的容量是小于数据总量的,所以,当缓存数据越来越多,Redis 不可避免的会被写满,这时候就涉及到 Redis 的内存淘汰机制了.我们需要选定某种策略将"不重要"的数据 ...

随机推荐

  1. 采用二进制文件方式安装loki和promtail

    1. 下载二进制文件 官方下载地址:https://github.com/grafana/loki/releases 下载如图所示的这俩文件,Grafana采用yum方式安装 cd /usr/loca ...

  2. 时序数据库之InfluxDB的基本操作

    1.进入Influxdb的客户端 [root@activity_sentinel ~]# influx 2.数据库的操作 显示所有的数据库名 > show databases name: dat ...

  3. Spring Data Jpa 更新操作

    第一步,通过Repository对象把实体根据ID查询出来 第二部,往查出来的实体对象进行set各个字段 第三步,通过Repository接口的save方法进行保存 保存和更新方式(已知两种) 第一种 ...

  4. 什么是B树(B-树)?

    写在开头:B-树,就是B树.因B树的英文名称为B-tree ,B-树因此而来,有人会误以为B-树是一种树,而B树又是另外一种树.实际上,B-tree就是指的B树. 而且B-树不可以读成B减树... 一 ...

  5. Linux 性能调优都有哪几种方法?

    1.Disabling daemons (关闭 daemons).    2.Shutting down the GUI (关闭 GUI).    3.Changing kernel paramete ...

  6. fiber核心(react 16)?

    旧: 浏览器渲染引擎单线程, 计算DOM树时锁住整个线程, 所有行为同步发生, 有效率问题, 期间react会一直占用浏览器主线程,如果组件层级比较深,相应的堆栈也会很深,长时间占用浏览器主线程, 任 ...

  7. 如何监控 Elasticsearch 集群状态?

    Marvel 让你可以很简单的通过 Kibana 监控 Elasticsearch.你可以实时查看你 的集群健康状态和性能,也可以分析过去的集群.索引和节点指标.

  8. java的API

    一.前端 1.jsp展示数据 (1)展示在前端控制台 console.table(参数); (2)弹窗 alert(参数); (3)JSLT的<c:if>标签 <c:if test= ...

  9. 什么是arduino及.arduino分类

    关于什么是arduino没有什么可说的,想要接触arduino多多少少的都会对arduino有一定的理解,我认为,arduino应该算是一个连接硬件与软件的平台,通过他,你可以将你的想法以代码的形式呈 ...

  10. HTML中meta标签详解;property=og标签详解

    meta是用来在HTML文档中模拟HTTP协议的响应头报文.META标签是HTML语言HEAD区的一个辅助性标签,它位于HTML文档头部的<HEAD>标记和<TITLE>标记之 ...