当前位置: 首页 > article >正文

golang学习笔记12——Go 语言内存管理详解

  • 推荐学习文档
    • golang应用级os框架,欢迎star
    • golang应用级os框架使用案例,欢迎star
    • 案例:基于golang开发的一款超有个性的旅游计划app经历
    • golang实战大纲
    • golang优秀开发常用开源库汇总
    • 想学习更多golang知识,这里有免费的golang学习笔记专栏

文章目录

    • 引言
    • Go 语言内存分配
      • 1.栈和堆
      • 2.内存分配器
      • 3.代码示例
    • Go 语言垃圾回收
      • 1.垃圾回收机制
      • 2.垃圾回收触发时机
      • 3.代码示例
    • Go 语言内存优化
      • 1.减少内存分配
      • 2.内存复用
      • 3.代码示例
    • 总结

引言

在软件开发中,内存管理是一个至关重要的问题。高效的内存管理可以提高程序的性能、减少内存泄漏和提高程序的稳定性。Go 语言作为一种现代编程语言,拥有一套先进的内存管理机制。本文将详细介绍 Go 语言的内存管理问题,包括内存分配、垃圾回收和内存优化等方面,并结合代码示例进行深入讲解。

Go 语言内存分配

1.栈和堆

  • 在 Go 语言中,内存被分为栈和堆两部分。栈用于存储局部变量和函数调用信息,而堆用于存储动态分配的内存。
  • 当一个函数被调用时,它的参数和局部变量会被分配在栈上。当函数返回时,这些变量会被自动释放。
  • 对于需要在函数调用之间保持存在的变量,或者动态分配的内存,会被分配在堆上。

2.内存分配器

  • Go 语言使用了一个高效的内存分配器来管理内存。这个分配器会根据对象的大小和生命周期选择合适的内存区域进行分配。
  • 对于小对象,分配器会尝试在栈上进行分配。如果栈上的空间不足,或者对象的生命周期较长,分配器会将对象分配在堆上。
  • 分配器还会对内存进行对齐,以提高内存访问的效率。

3.代码示例

package main

import "fmt"

func main() {
    // 局部变量分配在栈上
    var a int = 10
    fmt.Println(a)

    // 动态分配的内存分配在堆上
    b := new(int)
    *b = 20
    fmt.Println(*b)
}

在上面的代码中,局部变量 a 会被分配在栈上,而通过 new 函数分配的内存 b 会被分配在堆上。

Go 语言垃圾回收

1.垃圾回收机制

  • Go 语言拥有一个自动的垃圾回收机制,可以自动回收不再使用的内存。
  • 垃圾回收器会定期扫描内存,标记哪些对象是可达的(即正在被使用的),哪些对象是不可达的(即不再被使用的)。不可达的对象会被回收,释放它们占用的内存。
  • Go 语言的垃圾回收器采用了三色标记法和写屏障技术,以提高垃圾回收的效率和准确性。

2.垃圾回收触发时机

  • Go 语言的垃圾回收器会在以下情况下触发:
    • 堆内存的使用量达到一定阈值。
    • 一定时间间隔内没有进行垃圾回收。
    • 手动触发垃圾回收(通过调用 runtime.GC() 函数)。

3.代码示例

package main

import (
    "runtime"
    "time"
)

func main() {
    // 分配一些内存
    data := make([]int, 1000000)
    for i := range data {
        data[i] = i
    }

    // 手动触发垃圾回收
    runtime.GC()

    // 等待一段时间,让垃圾回收器有时间运行
    time.Sleep(time.Second)
}

在上面的代码中,首先分配了一个包含 1000000 个整数的切片,占用了一定的内存。然后手动触发了垃圾回收,最后等待一段时间,让垃圾回收器有时间运行。

Go 语言内存优化

1.减少内存分配

  • 在 Go 语言中,可以通过一些方法来减少内存分配,提高程序的性能。例如:
    • 尽量避免频繁地创建和销毁对象,可以重复使用对象或者使用对象池。
    • 对于小对象,可以考虑使用栈分配而不是堆分配。
    • 避免使用过大的切片或映射,尽量根据实际需求分配合适的大小。

2.内存复用

  • Go 语言中的一些数据结构,如切片和映射,可以进行内存复用。例如,可以使用 append 函数向切片中添加元素,而不是每次都创建一个新的切片。
  • 对于映射,可以使用 sync.Pool 来复用已经分配的内存,减少内存分配的次数。

3.代码示例

package main

import (
    "sync"
)

var pool = sync.Pool{
    New: func() interface{} {
        return make([]int, 0, 10)
    },
}

func main() {
    // 从对象池中获取一个切片
    slice := pool.Get().([]int)
    defer pool.Put(slice)

    // 向切片中添加元素
    slice = append(slice, 1, 2, 3)
    fmt.Println(slice)
}

在上面的代码中,使用了 sync.Pool 来复用已经分配的内存。首先定义了一个对象池,当需要一个切片时,可以从对象池中获取一个已经分配好的切片。使用完毕后,将切片放回对象池中,以便下次复用。

总结

Go 语言的内存管理机制使得开发者可以更加专注于业务逻辑的实现,而不必过多地关注内存管理的细节。通过合理地使用内存分配、垃圾回收和内存优化技术,可以提高程序的性能、减少内存泄漏和提高程序的稳定性。在实际开发中,开发者应该根据具体的需求选择合适的内存管理策略,以实现高效、可靠的程序。

关注我看更多有意思的文章哦!👉👉


http://www.kler.cn/a/305465.html

相关文章:

  • void * 指针与整数进行加减运算
  • Redis主从复制(replication)
  • git status 命令卡顿的排查
  • 第 4 章 - Go 语言变量与常量
  • Android HandlerThread 基础
  • Spring Boot集成SQL Server快速入门Demo
  • 软件测试工程师面试整理-数据库与SQL
  • 微信小程序使用 ==== 粘性布局
  • Ubuntu 常用指令和作用解析
  • 16. MyBatis的延迟加载机制是什么?如何配置?有哪些优缺点?
  • 股票程序化交易赚钱吗,证监会如何监测股票多账户关联交易
  • Vue 创建自定义指令 以及创建自定义指令相关属性说明
  • DFS 算法:洛谷B3625迷宫寻路
  • Redis相关命令详解
  • Understanding the model of openAI 5 (1024 unit LSTM reinforcement learning)
  • WSL安装Redis
  • 【linux】 cd命令
  • 代码随想录算法训练营第62天| 图论 Floyd算法 A*算法
  • 鸿蒙 NEXT 生态应用核心技术理念:可分可合,自由流转
  • 开源 AI 智能名片 S2B2C 商城小程序相关角色的探索
  • 基于vue框架的宠物爱好者交流网站的设计与实现p2653(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。
  • 黑马点评19——多级缓存-缓存同步
  • 基于SSM的银发在线教育云平台的设计与实现
  • Qt事件处理机制
  • wandb一直上传 解决方案
  • 大顶堆+动态规划+二分