网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
}
以上是一个libapr操作内存池的最简单的例子。上面的例子自然是没有内存泄漏的。
但是实际上的业务逻辑肯定不会像上面那么简单,而是形如下面的形式:
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <apr_general.h>
#include <apr_pools.h>
int got_exit = 0;
void term(int sig){
got_exit = 1;
}
int main(int argc, const char *argv[])
{
apr_status_t rv;
apr_pool_t *mp;
char *buf1;
char *buf2;
/\* 框架开始 \*/
rv = apr\_initialize();
if (rv != APR_SUCCESS) {
assert(0);
return -1;
}
/\* 1.创建内存池 \*/
apr\_pool\_create(&mp, NULL);
signal(SIGTERM, term);
signal(SIGINT, term);
/\* 2.业务逻辑 mainloop \*/
while(got_exit){
buf1 = apr\_palloc(mp, 1024);
buf2 = apr\_palloc(mp, 1024);
strcpy(buf1, "hello apr");
strcpy(buf2, buf1);
printf("buf1:%s, buf2:%s\n", buf1, buf2);
}
/\* 3.销毁内存池 \*/
apr\_pool\_destroy(mp);
/\* 框架结束 \*/
apr\_terminate();
return 0;
}
上面的代码更符合实际的项目代码使用逻辑。我们在程序一开始申请了一个内存池,然后进入mainloop进行业务处理,这个mainloop通常是不会退出的,只有当接收到了退出信号的时候,才会退出,在退出的时候,会销毁内存池。
这段代码的可怕之处在于,你使用valgrind之类的内存检测工具,根本检测不出来它有内存泄漏!但是随着程序的不间断运行,你的内存又会被程序不断蚕食,直到系统的物理内存全部被吃光!
而当代码变得庞杂之后,这样的细微操作是最不会让人提防的,很可能因为一个很小的函数里使用了内存池,而这个函数要被频繁调用,从而导致内存不断增加。所以,检查不出来的内存泄漏,才是最可怕的内存泄漏。
那么,这种情况就不能避免吗?
当然是可以的。我们需要养成任何时候都要使用子内存池的习惯,这样就不会造成无谓的内存池无限扩容,上述代码修改如下:
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <apr_general.h>
#include <apr_pools.h>
int got_exit = 0;
void term(int sig){
got_exit = 1;
}
int main(int argc, const char *argv[])
{
apr_status_t rv;
apr_pool_t *mp;
char *buf1;
char *buf2;
/\* 框架开始 \*/
rv = apr\_initialize();
if (rv != APR_SUCCESS) {
assert(0);
return -1;
}
/\* 1.创建内存池 \*/
apr\_pool\_create(&mp, NULL);
signal(SIGTERM, term);
signal(SIGINT, term);
/\* 2.业务逻辑 mainloop \*/
while(got_exit){
apr\_pool\_t \*child = NULL;
apr\_pool\_create(&child, mp);
buf1 = apr\_palloc(child, 1024);
buf2 = apr\_palloc(child, 1024);
strcpy(buf1, "hello apr");
strcpy(buf2, buf1);
printf("buf1:%s, buf2:%s\n", buf1, buf2);
apr\_pool\_destroy(child);
}
/\* 3.销毁内存池 \*/
apr\_pool\_destroy(mp);
/\* 框架结束 \*/
apr\_terminate();
return 0;
}
如上所示,我们每次都从父池子里申请一个子池子,然后使用完成后,调用apr\_pool\_destroy将子池子的内存归还给父池子,这样,只要每一轮调用中,没有特别出格的内存调用,父池子的大小会保持在一个相对稳定的大小,而不会无线扩张。
![img](https://img-blog.csdnimg.cn/img_convert/b484611af7abfee45c3af81ee6be0898.png)
![img](https://img-blog.csdnimg.cn/img_convert/951ba423348c2ff17ea3a25a1b87c550.png)
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618668825)**
C C++开发知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618668825)**