【C基础】堆内存创建/释放和内存清理函数/内存泄漏

本期涉及到了较多的指针,没有彻底领悟的同学请翻阅之前的博文~

一闪一闪亮晶晶,满天都是小星星***

什么是堆内存:

是进程的一个内存段(text、data、bss、heap、stack)之一,由程序员手动管理,
特点就是足够大,缺点就是使用麻烦。

为什么使用堆内存:

1、随着程序的复杂数量变多。
2、堆内存的申请释放受控制。

如何使用堆内存:

注意:C语言中没控制堆内存的语句,只能使用C标准库提供的函数。
#include <stdlib.h>void *malloc(size_t size);
功能:从堆内存中申请size个字节的内存,申请内存中存储是什么内容不确定。
返回值:成功返回申请到的内存的首地址,失败返回NULL。void free(void *ptr);
功能:释放一块堆内存,可以释放NULL,但不能重复释放和非法地址。
注意:释放仅仅是使用权,里面的数据不会被特意清理。

申请,释放和置空堆内存:

#include <stdio.h>
#include <stdlib.h>int main(int argc,const char* argv[])
{// 从堆上分配一块内存,首个字节的地址存放在指针变量p中int* p = malloc(sizeof(int)*10);for(int i=0; i<10; i++){p[i] = i;	printf("%d\n",p[i]);}// 解引用  使用堆内存*p = 1000;printf("%d\n",*p);// 释放堆内存free(p);// 重复释放会导致堆内存崩溃// free(p);// free只是释放使用权,数据不会被特意清理for(int i=0; i<10; i++){printf("%d\n",p[i]);}// 堆内存释放后指针要及时置空p = NULL;}
void *calloc(size_t nmemb, size_t size);
功能:从堆内存中申请nmemb块size个字节的内存,申请的内存块会被初始化为0。
注意:申请的依然是一块连续的内存。void *realloc(void *ptr, size_t size);
功能:改变已经有内存块的大小,在原有的基础上调大或调小。
返回值:是调整后的内存块的首地址,一定要重新接收返回值,可能不是在原内块上调整的。如果无法在原内存块上进行调整:1、申请一块新的符合要的内存块2、把原内存块上的内容拷贝过去3、把原内存释放掉返回新内存块的首地址
注:realloc函数有个争议性话题,对于大型工程可能存在内存泄漏,本人目前未能亲自验证,但不可不防。
#include <stdio.h>
#include <stdlib.h>int main(int argc,const char* argv[])
{int* p = malloc(40);for(int i=0; i<10; i++){p[i] = i;printf("%d ",p[i]);}malloc(20);int* p1 = realloc(p,80);printf("%p %p\n",p,p1);for(int i=0; i<10; i++){p1[i] = i;printf("%d ",p1[i]);}}

malloc的内存管理机制:

当首次向malloc申请内存时,malloc会向操作系统申请内存,操作系统会直接分配33页(1页=4096字节)内存交给malloc管理。但这不意味着可以越界访问,因为malloc把使用分配给"其他人",这样会产生脏数据。每个内存块之间会有些空隙(4~12字节),这些空隙一些是为了内容对齐,其中有4个字节记录着malloc维护信息,这些维护信息决定下次分配内存的位置,也可以借助计算出每个内存块的大小,当这些信息被破坏时就会影响malloc和free函数的调用。

使用堆内存要注意的问题:

内存泄漏:内存无法再使用,也无法被释放,而再次使用时只能重新申请,然后重复心以上过程,是积月累导致系统中可用的内存越来越少。注意:程序一旦结束属于它的资源都会被操作系统回收。谁申请谁释放,谁知道该释放谁释放。如何判断内存定位泄漏:1、查看内存的使用情况。2、使用代码分析工具检查malloc的调用情况。3、包装malloc、free,记录申请、释放的信息到日志中。内存碎片:已经释放的但无法再继续使用的内存叫内存碎片,是由于申请和释放的时间不协调导致的,无法避免只能尽量减少。如何减少内存碎片:1、尽量使用栈内存。2、不要频繁的申请释放内存。3、尽量申请大块内存自己管理。

内存清理函数:

#include <strings.h>void bzero(void *s, size_t n)
功能:把一块内存清理为0
s:内存块的首地址
n:内存块的字节数#include <string.h>
void *memset(void *s, int c, size_t n);
功能:把内存块按字节设置为c
s:内存块的首地址
c:相设置的ASCII码值
n:内存块的字节数
返回值:设置成功后的内存首地址
#include <stdio.h>
#include <stdlib.h>
#include <strings.h>int main(int argc,const char* argv[])
{int* p = malloc(40);for(int i=0; i<10; i++){p[i] = i;printf("%d\n",p[i]);}//bzero(p,40);memset(p,1,40);printf("---------------\n");for(int i=0; i<10; i++){printf("%d\n",p[i]);}
}

堆内存定义二维数组:

指针数组:定义n*m二维数组类型* arr[n];for(int i=0; i<n; i++){arr[i] = malloc(sizeof(类型)*m);}注意:每行的m的值可以不同,这就是不规则二维数组。
数组指针:类型 (*arr)[n] = malloc(类型*n*m);注意:所谓的多维数组都是用一维数组模拟的。
使用示例:

1、

#include <stdio.h>
#include <stdlib.h>int main(int argc,const char* argv[])
{int* arr[10] = {};for(int i=0; i<10; i++){arr[i] = malloc(40);}for(int i=0; i<10; i++){for(int j=0; j<10; j++){printf("%d ",arr[i][j]);}printf("\n");}
}

2、

#include <stdio.h>
#include <stdlib.h>int main(int argc,const char* argv[])
{int (*arr)[5] = malloc(sizeof(int)*4*5);for(int i=0; i<4; i++){for(int j=0; j<5; j++){printf("%d ",arr[i][j]);}printf("\n");}
}

3、

#include <stdio.h>int main(int argc,const char* argv[])
{int arr[4][5] = {0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19};//int *p = (int*)arr;int (*p)[5] = arr;for(int i=0; i<4; i++){for(int j=0; j<5; j++){printf("%d ",p[i][j]);}printf("\n");}
}
练习1:计算出 100~10000 之间的素数,结果存储在堆内存中,不要浪费内存。
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <math.h>bool is_prime(int num)
{for(int i=2;i<=sqrt(num);i++){if(0 == num % i)return false;}return true;
}int main(int argc,char* argv[])
{int* arr=NULL,cnt=0;for(int i=100; i<1000;i++){if(is_prime(i)){arr = realloc(arr,sizeof(int)*(cnt+1));arr[cnt++]=i;}}for(int i=0;i<cnt;i++){printf("%d ",arr[i]);}free(arr);arr=NULL;
}

字符:

在计算机中字符是以整数形式存储在,当需要显示时会根据ASCII码表中的对应关系显示出相应的符号或图案。
'\0' 0
'0' 48
'a' 97
'A' 65字符的输入:scanf("%c",&ch);ch = getchar();
字符的输出:printf("%c",ch);putchar(ch);

串:

是一种数据结构,由一组连续的若干个类型相同的数据组成,有一个结束标志。

字符串:

由字符组成的串型结构,结束标志就是'\0'。字符串存在的形式:
字符串字面值:"由双引号包含的若干个字符",以地址形式存在,数据存储在代码段,如果修改就会产生段错误。const char* str = "字面串字面值";sizeof("strstr") 结果字符个数加1。注意:它的结束标志隐藏在末尾。两个一模一样的字符串字面值在代码段中只有一份。字符数组:由char类型组成的数组,要为'\0'预留位置。使用的栈内存,数据可以修改。常用方式:字符数组[] = "字符串字面值";会自动为'\0'预留位置。    赋值完成后字符串存在两份,一份存储在代码段,另一份存储在栈内存(可被修改)。

字符串的输入:

scanf %s 地址 缺点:不能输入空格char* gets(char *s);
功能:输入字符串,并且可以接收空格
返回值:链式调用(把一个函数的返回值,作业另一个函数的参数)。char *fgets(char *s, int size, stdin);
功能:可以设置输入的字符串的长度size-1字符,超出的部分不接收,它会为'\0'预留位置。
注意:如果输入的字符数不足size-1,最后的\n会一起接收了。

字符串的输出:

printf %s 地址int puts(const char *s);
功能:输出一个字符串,会在末尾自动添加一个\n。
返回值:成功输出的字符个数。

练习2:实现一个判断一个字符串是否是回文串。

#include <stdio.h>
#include <stdbool.h>bool is_palindrome(const char* str)
{size_t len = 0;/* 获取字符串长度 */while(str[len]) len++;for(int i=0; i<len/2; i++){if(str[i] != str[len-i-1]) return false;}return true;
}int main(int argc,const char* argv[])
{char str[255] = {};printf("%s\n",is_palindrome(gets(str))?"是回文串":"不是回文串");
}

练习3:实现一个函数把一个由数字字符组成的字符串转换成整数。

#include <stdio.h>int str_to_int(const char* str)
{int num = 0;while(*str && '0'<=*str && *str<='9'){num = num*10 + *str-'0';str++;}return num;
}int main(int argc,const char* argv[])
{char str[256] = {};printf("%d\n",str_to_int(gets(str)));
}

练习4:实现一个函数把一个字符串逆序。

#include <stdio.h>char* reverse(char* str)
{size_t len = 0;while(str[len]) len++;for(int i=0; i<len/2; i++){char t = str[i];str[i] = str[len-i-1];str[len-i-1] = t;}return str;
}int main(int argc,const char* argv[])
{char str[255] = {};puts(reverse(gets(str)));
}

[OVER]

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

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

相关文章

19_05_01校内训练[polygon]

题意 把一个边长为1的正n边形放到一个正m边形中&#xff0c;要求m边形完全覆盖n边形&#xff0c;可以有交点&#xff0c;并且中心重合。求正m边形的最小边长&#xff0c;至少精确到6位。要求logn计算。 思考 先考虑m|n的情况。 我们知道&#xff0c;正m边形的边长与可行区域&am…

六度人脉 全球最高效的人脉法则(图)

六度人脉这一概念&#xff0c;在20世纪60年代由美国心理学家Stanley Milgram提出并验证。 所谓六度人脉&#xff0c;即地球上所有的人都可以通过六层以内的熟人关系链和其他人联系起来。 通俗地说&#xff1a;“最多通过六个人你就可以认识地球上任何一个陌生人。” SNS(社会…

[转]numpy中的np.max 与 np.maximum区别

转自&#xff1a;https://blog.csdn.net/lanchunhui/article/details/52700895 转载于:https://www.cnblogs.com/xianhan/p/10609319.html

JVM 的 Finalization Delay 引起的 OOM(java.lang.OutOfMemoryError:null at sun.misc.Unsafe.allocateMemory.)

今天在压力测试环境某一个服务出现crash了&#xff0c;经过一番检查&#xff0c;终于发现是由于JVM的Finalization Delay引起的&#xff0c;这个问题比较特殊&#xff0c;这里记录一下。 这个服务是用Java写的&#xff0c;主要完成的功能是根据特定的指令文件生成mp4文件&#…

win10 php7+apache2.4的配置以及遇到的问题及解决

首先进入PHP官网下载php7的版本,我下的是PHP7.1.28,在PHP的下载页面注意划红线和绿线的地方(我画的) 1.画了红线的意思是请使用由apache lounge提供的编译文件,也就是点进蓝色Apache lounge这里下载. 2.画了绿色的线的意思是用Apache的话你必须使用Thread Safe(线程安全)的PHP…

缓存区的输入输出,字符串常用操作,实现strlen/strcpy/strcat/strcmp函数)

输出缓冲区&#xff1a; 程序输入的数据并不能立即显示在屏幕上&#xff0c;而是先存储在输出缓冲区中&#xff0c;满足一些条件后才显示出来。 1、遇到\n后 2、遇到输入语句 3、当输出缓冲区满4K 4、当程序结束 5、手动刷新 fflush(stdout) 缓冲区机制可以提高数据的读写速度…

理性分散投资 收益袋袋平安

理财锦囊 想要投资理财&#xff0c;不光可以选择股票和债券这类入门产品&#xff0c; 实际上&#xff0c;还可选择其他低风险及高回报的投资产品&#xff0c;例如外汇、期货和商品。 针对此&#xff0c;几位分析师预测了2014年各国经济走势的重点&#xff0c;协助散户们分配…

AI一周热闻:华为豪掷3.3亿剑桥买地,自建光芯片工厂;比特大陆IPO失败,组织架构调整...

导读 华为豪掷3.3亿剑桥买地&#xff0c;自建光芯片工厂苹果春季发布会无硬件发布&#xff0c;转型之心迫切比特大陆IPO失败&#xff0c;组织架构调整&#xff0c;王海超任CEO特斯拉起诉小鹏汽车员工窃取商业机密英伟达发布GauGAN&#xff0c;线条色块秒变逼真图像用机器学习防…

Docker 环境:Nexus3.x 的私有仓库

Nexus3.x 的私有仓库 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 使用 Docker 官方的 Registry 创建的仓库面临一些维护问题。比如某些镜像删除以后空间默认是不会回收的&#xff…

虚拟环境vitualenv的使用

Python3开发之虚拟环境virtualenv与virtualenvwrapper 在使用 Python 开发的过程中&#xff0c;工程一多&#xff0c;难免会碰到不同的工程依赖不同版本的库的问题&#xff1b; 亦或者是在开发过程中不想让物理环境里充斥各种各样的库&#xff0c;引发未来的依赖灾难。 此时&am…

find_first_of和find函数的区别

小记&#xff1a; find_first_of函数最容易出错的地方是和find函数搞混。它最大的区别就是如果在一个字符串str1中查找另一个字符串str2&#xff0c;如果str1中含有str2中的任何字符&#xff0c;则就会查找成功&#xff0c;而find则不同&#xff1b;

银行各类理财收益渐涨 各类宝钱景尚不明朗

这个春天&#xff0c;投资似乎进入了一个好事多磨的阶段。央行一反先前支持的态度&#xff0c;开始对互联网理财念起了“紧箍咒”。一时间&#xff0c;各种“宝”的命运变得扑朔迷离起来。尽管各种“宝”声明&#xff1a;不受央行政策影响。而投资者内心的担忧&#xff0c;恐怕…

Firefox 66回归!修复多项臭虫相关问题

上周最新版Firefox 66因为爆出会使微软Office 365中的PowerPoint文字消失的臭虫&#xff0c;Mozilla暂停发送。3月27日Mozilla重新释出修补完成的最新版Firefox 66.0.2。根据Mozilla臭虫报告网页&#xff0c;Firefox 66除了造成Office 365中的PowerPoint文字消失的问题外&#…

PHP全栈学习笔记27

数组概述&#xff0c;类型&#xff0c;声明&#xff0c;遍历&#xff0c;输出&#xff0c;获取数组中最后一个元素&#xff0c;删除重复数组&#xff0c;获取数组中指定元素的键值&#xff0c;排序&#xff0c;将数组中的元素合成字符串。 数组概述&#xff0c;数组是存储&…

Docker : 数据卷(创建、挂载、查看、删除)

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 数据卷 数据卷 是一个可供一个或多个容器使用的特殊目录&#xff0c;它绕过 UFS&#xff0c;可以提供很多有用的特性&#xff1a; 数据卷…

mac地址和ip地址的区别(转)

先纠正一下几个比较模糊的概念&#xff1a;“MAC地址表储存IP地址”&#xff0c; MAC地址表是二层设备中存储“MAC地址”和“转发端口”映射关系的表&#xff0c;并不直接存储IP地址。 “路由器根据MAC地址来选择路由进行数据发送”&#xff0c;对于三层设备的三层端口来说&…

你是否发现 职业能力危机,请 警惕

身在职场&#xff0c;你有不有遭遇职业能力危机呢 ? 核心竞争力的增长是职业持续性发展的基础&#xff0c;随着年龄的增长和工作经验的积累&#xff0c;有的职场人士保持着良好的发展势态&#xff0c;有的却越来越落伍&#xff0c;竞争力越来越弱。只有能力跟得上变化&#x…

你的GitHub,怎么和我用的不太一样?

说起代码托管&#xff0c;相信绝大多数人脑海中浮现出的第一个词都是“GitHub”。经过多年的发展&#xff0c;GitHub俨然已经成为了代码托管领域的标签…随着国内互联网环境的优化&#xff0c;互联网产业链的不断壮大&#xff0c;越来越多的产业被拉入到了互联网中来&#xff0…

Windows下多个JDK版本的切换方法

问题 因我之前在window中无法命令行输入&#xff0c;后来发现是电脑中存在多个JDK&#xff0c;导致设置混乱。于是&#xff0c;我继续深入研究了当电脑存在多个JDK的情况下&#xff0c;如何设置想要的JDK版本。步骤 1.更改环境变量 进入环境变量的配置界面&#xff0c;将JAVA_H…

哈哈哈,只有程序员才懂的黑色幽默 ... ...

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 也是机缘巧合&#xff0c;让我一个之前工作从未接触过程序员的人&#xff0c;现在成天和程序员打交道&#xff0c;要知道&#xff0c;不…