Go运行时的内存分配器以及消耗指定大小的内存(C语言)

news/2024/5/7 20:24:19/文章来源:https://blog.csdn.net/weixin_41896770/article/details/128035486

对于go语言在运行时的一些内存分配,想要详细的了解,我们会用到自带的runtime.MemStats,有很多具体的细节实现,而不是简单的只看任务管理器中的内存分配。

我们先来看下这个记录内存分配器的结构体

type MemStats struct {Alloc uint64		#堆空间分配的字节数TotalAlloc uint64		#服务运行一直累积的总分配的堆空间,释放也不减少Sys uint64		#操作系统获取的内存Lookups uint64		#运行时执行的指针查找数,用于调试Mallocs uint64		#服务malloc的次数Frees uint64		#释放堆对象字节数HeapAlloc uint64		#服务分配的堆内存字节数HeapSys uint64		#系统分配的堆内存字节数HeapIdle uint64		#申请但未分配的堆内存或者回收了的堆内存字节数HeapInuse uint64		#正在使用的堆内存字节数HeapReleased uint64	#返回给OS的堆内存,类似C/C++中的freeHeapObjects uint64	#堆内存块申请的量StackInuse uint64		#正在使用的栈字节数StackSys uint64		#系统分配的作为运行栈的内存MSpanInuse uint64		#用于测试用的结构体使用的字节数MSpanSys uint64		#系统为测试用的结构体分配的字节数MCacheInuse uint64	#结构体申请的字节数,不被视为垃圾回收	MCacheSys uint64		#操作系统申请的堆空间用于mcache的量BuckHashSys uint64	#用于剖析桶散列表的堆空间GCSys uint64		#垃圾回收标记元信息使用的内存OtherSys uint64		#golang系统架构占用的额外空间NextGC uint64		#垃圾回收器检视的内存大小LastGC uint64		#垃圾回收器最后一次执行时间PauseTotalNs uint64	#垃圾回收或者其他信息收集导致服务暂停的次数PauseNs [256]uint64	#一个循环队列,记录最近垃圾回收系统中断的时间PauseEnd [256]uint64	# 一个循环队列,记录最近垃圾回收系统中断的时间开始点NumGC uint32		#垃圾回收完成的次数NumForcedGC uint32	#服务调用runtime.GC()强制使用垃圾回收的次数// GODEBUG=gctrace=1.GCCPUFraction float64	#垃圾回收占用服务CPU工作的时间总和#如果有100个goroutine,垃圾回收的时间为1s,总占用100s	EnableGC bool		#是否启用GC,如果启用,即便GOGC=off也是启用的DebugGC bool#内存分配器使用情况BySize [61]struct {Size uint32Mallocs uint64Frees uint64}
}

看个具体的示例:

package mainimport ("fmt""runtime""time"
)// 内存使用情况
func PrintMemUsage() {var m runtime.MemStatsruntime.ReadMemStats(&m)fmt.Printf("Alloc = %v MiB", bToMb(m.Alloc))fmt.Printf("\tTotalAlloc = %v MiB", bToMb(m.TotalAlloc))fmt.Printf("\tSys = %v MiB", bToMb(m.Sys))fmt.Printf("\tNumGC = %v\n", m.NumGC)
}// 转国际标(兆字节)
func bToMb(b uint64) uint64 {return b / 1024 / 1024
}
func main() {var s []stringfor i := 0; i < 100000000; i++ {s = append(s, "你好,寅恪光潜!")}PrintMemUsage()for {//runtime.GC() //调用GC进行内存回收PrintMemUsage()fmt.Println(s[0])time.Sleep(time.Second * 1)}
}
/*
C:\Users\Tony>go run test.go
Alloc = 5625 MiB        TotalAlloc = 9192 MiB   Sys = 8066 MiB  NumGC = 25
Alloc = 4486 MiB        TotalAlloc = 9192 MiB   Sys = 8066 MiB  NumGC = 26
你好,寅恪光潜!
Alloc = 4486 MiB        TotalAlloc = 9192 MiB   Sys = 8066 MiB  NumGC = 26
你好,寅恪光潜!
Alloc = 4486 MiB        TotalAlloc = 9192 MiB   Sys = 8066 MiB  NumGC = 26
你好,寅恪光潜!
Alloc = 4486 MiB        TotalAlloc = 9192 MiB   Sys = 8066 MiB  NumGC = 26
...
*/

然后去掉runtime.GC()注释,启用内存回收,出现如下效果:

/*
C:\Users\Tony>go run test.go
Alloc = 4486 MiB        TotalAlloc = 9192 MiB   Sys = 7050 MiB  NumGC = 26
Alloc = 1838 MiB        TotalAlloc = 9192 MiB   Sys = 7050 MiB  NumGC = 27
你好,寅恪光潜!
Alloc = 1838 MiB        TotalAlloc = 9192 MiB   Sys = 7050 MiB  NumGC = 28
你好,寅恪光潜!
Alloc = 1838 MiB        TotalAlloc = 9192 MiB   Sys = 7050 MiB  NumGC = 29
你好,寅恪光潜!
Alloc = 1838 MiB        TotalAlloc = 9192 MiB   Sys = 7050 MiB  NumGC = 30
...
*/

上面选取了几个成员变量,可以看出这个运行时的内存查看还是挺方便详细的。

有时候需要做内存压力测试等,我们可能会用到消耗一定的内存,没有Linux纯环境,我在WSL中运行,如果你也是Windows系统想要熟悉Linux也可以查阅:Windows版的Linux子系统(WSL)安装

malloc_mb.c

#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>#define UNIT (1024*1024)int main(int argc, char *argv[])
{long long i = 0;int size = 0;if (argc != 2){printf("请输入需要分配的内存值\n");return 1;}#unsigned long int strtoul(const char *nptr, char **endptr, int base);#unsigned long long int strtoull(const char *nptr, char **endptr,int base);size = strtoull(argv[1], NULL, 10);#字符串转换成数字,十进制if (size == 0){printf("请输入大于0的内存值");return 1;}char *buff = (char *) malloc(size * UNIT);if (buff)printf("已分配%dMB\n", size);buff[0] = 1;for (i = 1; i < (size * UNIT); i++){if (i%1024 == 0)buff[i] = buff[i-1]/8;elsebuff[i] = i/2;}pause();
}

编译成一个可执行文件:gcc malloc_mb.c -o malloc,目录下面将有个malloc文件,然后运行它:./malloc 1000
这样就会成功分配1000MB的内存!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_225697.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

一文了解 Go 中的指针和结构体

一文了解 Go 中的指针和结构体前言指针指针的定义获取和修改指针所指向变量的值结构体结构体定义结构体的创建方式小结耐心和持久胜过激烈和狂热。 前言 前面的两篇文章对 Go 语言的基础语法和基本数据类型以及几个复合数据类型进行介绍&#xff0c;本文将对 Go 里面的指针和结…

MySQL索引底层数据结构

索引简介 索引是一个排好序的数据结构&#xff0c;包含着对数据表里所有记录的引用指针&#xff0c;如下图所示。索引文件和数据文件一样都存储在磁盘中&#xff0c;数据库索引的目的是在检索数据库时&#xff0c;减少磁盘读取次数。 常见的索引数据结构包括二叉树、红黑树、…

跬智信息 (Kyligence) 荣获信创“大比武”重要奖项,坚持做大做实国产软件

近日&#xff0c;为期两个月的 2022 信创“大比武”活动圆满闭幕。经过层层筛选和考核&#xff0c;跬智信息 (Kyligence) 凭借“企业级智能多维数据分析解决方案”项目脱颖而出&#xff0c;在整体方案的技术架构、服务体系、安全架构、信创生态等方面得到了评委的高度认可&…

iptables应用大全

iptables四表五链&#xff1a; 1、“四表”是指 iptables 的功能 ——filter 表&#xff08;过滤规则表&#xff09;&#xff1a;控制数据包是否允许进出及转发 ——nat 表&#xff08;地址转换规则表&#xff09;&#xff1a;控制数据包中地址转换 ——mangle&#xff08;修改…

NDK 是什么 | FFmpeg 5.0 编译 so 库

前言 NDK 全称 Native Development Kit&#xff0c;也就是原生开发工具包 &#xff0c;官网对它有详细的 中文介绍 。可能一说到 NDK 或 JNI &#xff0c;大家脑子里第一反应就是集成 C/C 。其实 JNI 的含义是 Java Native Interface &#xff0c;这种接口允许 Java 和其他语言…

ovs vxlan 时延和吞吐

设计云时到底要不要用vxlan&#xff0c;如果用vxlan到底要不要购买比较贵的smart nic做offload&#xff0c;采用软件vxlan还是硬件交换机vxlan&#xff0c;很难决策&#xff0c;这儿简单测试一下&#xff0c;给个参考&#xff0c;资源终究是有限的&#xff0c;成本还是有考虑的…

【HDU No. 2586】 树上距离 How far away ?

【HDU No. 2586】 树上距离 How far away &#xff1f; 杭电 OJ 题目地址 【题意】 有n 栋房屋&#xff0c;由一些双向道路连接起来。 每两栋房屋之间都有一条独特的简单道路&#xff08;“简单”意味着不可以通过两条道路去一个地方&#xff09;。人们每天总是喜欢这样问&a…

Linux 软链接 与 硬链接 的区别

Linux 软链接 与 硬链接 的区别 1、概念 ​  链接文件&#xff1a;是 Linux 操作系统中的一种文件&#xff0c;主要用于解决文件的共享使用问题&#xff0c;而链接的方式分为两种——软链接和硬链接。 ​  inode&#xff1a;是文件系统中存储文件元信息&#xff08;文件的…

3.71 OrCAD新建原理图时,每一个类目的含义是什么?OrCAD软件怎么显示元器件的封装名称?

笔者电子信息专业硕士毕业&#xff0c;获得过多次电子设计大赛、大学生智能车、数学建模国奖&#xff0c;现就职于南京某半导体芯片公司&#xff0c;从事硬件研发&#xff0c;电路设计研究。对于学电子的小伙伴&#xff0c;深知入门的不易&#xff0c;特开次博客交流分享经验&a…

Word处理控件Aspose.Words功能演示:在 Python 中将 Word 文档转换为 PNG、JPEG 或 BMP

MS Word 文件到图像格式的转换让您可以将文档的页面嵌入到您的 Web 或桌面应用程序中。为了在 Python 应用程序中执行此转换&#xff0c;本文介绍了如何使用 Python 将 Word DOCX或DOC文件转换为PNG、JPEG或BMP图像。此外&#xff0c;您将学习如何使用不同的选项控制 Word 到图…

SpringBoot2.7.4整合Redis

目录 一、添加maven依赖 二、添加配置项 三、新增配置类 四、编辑实体类 五、编写接口 六、编写业务层 1.编写service层 2.编写service实现层 七、测试接口 一、添加maven依赖 <dependency><groupId>org.springframework.boot</groupId><artif…

Python测试框架之Pytest基础入门

Pytest简介 Pytest is a mature full-featured Python testing tool that helps you write better programs.The pytest framework makes it easy to write small tests, yet scales to support complex functional testing for applications and libraries. 通过官方网站介绍…

Flink部署之Yarn

Flink部署之Yarn 一、环境准备 1、Flink 是一个分布式的流处理框架&#xff0c;所以实际应用一般都需要搭建集群环境。 需要准备 3 台 Linux 机器。具体要求如下&#xff1a; 系统环境为 CentOS 7.5 版本。安装 Java 8。安装 Hadoop 集群&#xff0c;Hadoop 建议选择 Hadoop…

【代码随想录】二刷-二叉树

# 二叉树《代码随想录》 二叉树的遍历方式 深度优先遍历: 前序遍历(递归法、迭代法): 中左右中序遍历(递归法、迭代法): 左中右后序遍历(递归法、迭代法): 左右中 广度优先遍历: 层序遍历(迭代法) 二叉树的定义 struct TreeNode{int val;TreeNode* left;TreeNode* right;Tree…

React - Ant Design3.x版本安装使用,并按需引入和自定义主题

React - Ant Design3.x版本安装使用&#xff0c;并按需引入和自定义主题一. 安装使用 antd二&#xff0e;antd 高级配置安装 react-app-rewired&#xff0c;对 create-react-app 的默认配置进行自定义安装 babel-plugin-import &#xff0c;按需加载组件代码和样式自定义主题An…

[毕业设计]机器学习水域检测标注算法

前言 &#x1f4c5;大四是整个大学期间最忙碌的时光,一边要忙着备考或实习为毕业后面临的就业升学做准备,一边要为毕业设计耗费大量精力。近几年各个学校要求的毕设项目越来越难,有不少课题是研究生级别难度的,对本科同学来说是充满挑战。为帮助大家顺利通过和节省时间与精力投…

IO模型Netty

一、IO模型 对于一次IO操作&#xff0c;数据会先拷贝到内核空间中&#xff0c;然后再从内核空间拷贝到用户空间中&#xff0c;所以一次read操作&#xff0c;会经历以下两个阶段&#xff0c;基于这两个阶段就产生了五种不同的IO模式。 为了避免用户进程直接操作内核&#xff0c;…

Android8.1 MTK 浏览器下载的apk点击无反应不能安装

最近测试人员发现用原生浏览器下载的apk点击安装时无反应&#xff0c;不能安装。 在/vendor/mediatek/proprietary/packages/apps/Browser/src/com/android/browser/DownloadHandler.java 中&#xff0c;发现下载的apk文件缺少了mime类型&#xff0c;如下图 mimetype null造…

RS编码译码误码率性能matlab仿真

目录 1.算法描述 2.仿真效果预览 3.MATLAB部分代码预览 4.完整MATLAB程序 1.算法描述 纠错编码技术在卫星通信、移动通信及数字存储等领域已获得了广泛的应用。RS码作为其中最重要的码类之一,具有优良的纠随机错误和突发错误的能力,被空间数据系统咨询委员会(CCSDS)作为一种…

计算机毕业设计——基于SpringBoot框架的网上购书系统的设计与实现

文章目录前言一、背景及意义选题背景选题目的二、系统设计主要功能运行环境三、系统实现部分页面截图展示部分代码展示四、源码获取前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 二十一世纪是网络化&#xff0c;信息化的时代&#xff0c;为了满足广大…