【动态内存管理助力程序优化与性能飞升】(中)

简介: 【动态内存管理助力程序优化与性能飞升】

【动态内存管理助力程序优化与性能飞升】(上):https://developer.aliyun.com/article/1424811


3. 常见的动态内存错误


3.1 对NULL指针的解引用操作


void test()
{
  int* p = (int*)malloc(INT_MAX / 4);
    //malloc函数开辟失败就会返回NULL
  *p = 20;//如果p的值是NULL,就会有问题
  free(p);
}


3.2 对动态开辟空间的越界访问


void test()
{
  int i = 0;
  int* p = (int*)malloc(10 * sizeof(int));
  if (NULL == p)
  {
    exit(EXIT_FAILURE);
  }
  for (i = 0; i <= 10; i++)
  {
    *(p + i) = i;//当i是10的时候越界访问
  }
  free(p);
}


3.3 对非动态开辟内存使用free释放


void test()
{
  int a = 10;
  int* p = &a;
  free(p);//ok?
}


3.4 使用free释放一块动态开辟内存的一部分


void test()
{
  int* p = (int*)malloc(100);
  p++;
  free(p);//p不再指向动态内存的起始位置
}


3.5 对同一块动态内存多次释放


void test()
{
  int* p = (int*)malloc(100);
  free(p);
  free(p);//重复释放
}


3.6 动态开辟内存忘记释放(内存泄漏)


void test()
{
  int* p = (int*)malloc(100);
  if (NULL != p)
  {
    *p = 20;
  }
}
int main()
{
  test();
  while (1);
}


忘记释放不再使用的动态开辟的空间会造成内存泄漏。


切记:动态开辟的空间一定要释放,并且正确释放 。


4. 几个经典的笔试题


demo1:


#include<stdio.h>
#include<stdlib.h>
#include<string.h>
void GetMemory(char* p)
{
  p = (char*)malloc(100);
}
void Test(void)
{
  char* str = NULL;
  GetMemory(str);
  strcpy(str, "hello world");
  printf(str);
}
int main()
{
  Test();
  return 0;
}


问题:对NULL解引用以及没有释放malloc申请的空间



解释:

GetMemory 函数: 这个函数接受一个字符指针(char*)作为参数,并尝试使用 malloc 来分配 100 字节的内存。然而,需要理解的是,在C语言中,函数参数是通过值传递的,这意味着 GetMemory 函数内部的 p 是从 Test 函数传递过来的指针的一个拷贝。对于拷贝的指针所做的更改不会影响 Test 函数中的原始指针。


Test 函数: 在 Test 函数中,声明并初始化了一个 char* 变量 str,并将其设置为 NULL。然后,调用 GetMemory 函数,并将 str 作为参数传递进去。由于参数是通过值传递的,GetMemory 函数只会修改它自己的指针拷贝,并不会改变 Test 函数中的原始 str 指针。


内存分配问题: 在 GetMemory 函数内部,内存被分配给局部指针 p,这是从 Test 函数的 str 指针拷贝过来的。这意味着在 Test 函数中,原始的 str 指针仍然是 NULL,并没有被赋予新分配的内存地址。


缓冲区溢出: 在调用 GetMemory 函数后,有一个 strcpy 函数调用,试图将字符串 "hello world" 复制到 str 指针中。然而,由于 str 仍然是 NULL(没有指向已分配的内存),这将导致未定义行为,并可能导致段错误或其他错误,因为访问了无效的内存。


修改:


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void GetMemory(char** p)
{
    *p = (char*)malloc(100); // 分配 100 字节内存,并将地址存储在原始指针中
}
void Test(void)
{
    char* str = NULL;
    GetMemory(&str); // 传递指向指针的指针(双重指针),以修改原始指针
    strcpy(str, "hello world");
    printf("%s", str);
    free(str); // 使用完内存后别忘了释放它
    str == NULL;
}
int main()
{
    Test();
    return 0;
}


demo2:


#include<stdio.h>
char* GetMemory(void)
{
  char p[] = "hello world";
  return p;//返回局部变量的地址
}
void Test(void)
{
  char* str = NULL;
  str = GetMemory();
  printf(str);
}
int main()
{
  Test();
  return 0;
}


问题:返回局部变量的地址


解释:

  1. GetMemory 函数: 这个函数声明了一个字符数组 p 并初始化为 "hello world"。然后它试图返回 p 的地址。但是,需要注意的是,p 是一个局部变量,它在函数结束时会被销毁。因此,将局部变量的地址返回给调用者是不安全的,因为在调用者函数中访问返回的地址将指向无效的内存区域。


  1. Test 函数: 在 Test 函数中,声明了一个字符指针 str 并将其初始化为 NULL。然后,调用 GetMemory 函数,将返回的地址赋值给 str。


  1. 错误的返回局部变量地址: 在 GetMemory 函数中,由于返回局部变量 p 的地址,str 指针现在指向了一个不再有效的内存地址,因为 p 在 GetMemory 函数返回后已经被销毁。


  1. printf 函数: 在 printf 中尝试打印 str 指向的字符串时,由于 str 指向无效内存地址,代码的行为将是未定义的。这可能导致程序崩溃、输出奇怪的字符或其他不确定的结果。


修改:


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
char* GetMemory(void)
{
    char* p = (char*)malloc(12); // 在堆上分配内存以容纳 "hello world" 和空结束符
    strcpy(p, "hello world"); // 将 "hello world" 复制到新分配的内存块中
    return p; // 返回指向分配内存的指针
}
void Test(void)
{
    char* str = NULL;
    str = GetMemory();
    printf("%s", str);
    free(str); // 使用完内存后别忘了释放它
}
int main()
{
    Test();
    return 0;
}


demo3:


#include<stdio.h>
#include<stdlib.h>
#include<string.h>
void GetMemory(char** p, int num)
{
  *p = (char*)malloc(num);
}
void Test(void)
{
  char* str = NULL;
  GetMemory(&str, 100);
  strcpy(str, "hello");
  printf(str);
}
int main()
{
  Test();
  return 0;
}


问题:malloc申请的空间没有释放


修改:


#include<stdio.h>
#include<stdlib.h>
#include<string.h>
void GetMemory(char** p, int num)
{
  *p = (char*)malloc(num);
}
void Test(void)
{
  char* str = NULL;
  GetMemory(&str, 100);
  strcpy(str, "hello");
  printf(str);
    free(str);
    str = NULL;
}
int main()
{
  Test();
  return 0;
}


【动态内存管理助力程序优化与性能飞升】(下):https://developer.aliyun.com/article/1424821

相关文章
|
1月前
|
机器学习/深度学习 算法 PyTorch
125_训练加速:FlashAttention集成 - 推导注意力优化的独特内存节省
2025年,大型语言模型的训练面临着前所未有的挑战。随着模型参数量和序列长度的不断增加,传统注意力机制的内存瓶颈问题日益突出。FlashAttention作为一种突破性的注意力算法,通过创新的内存访问模式和计算优化,显著提升了训练效率和内存利用。
|
1月前
|
存储 机器学习/深度学习 PyTorch
119_LLM训练的高效内存管理与优化技术:从ZeRO到Flash Attention
大型语言模型(LLM)的训练面临着前所未有的计算和内存挑战。随着模型规模达到数百亿甚至数千亿参数,高效的内存管理成为训练成功的关键因素之一。2025年,LLM训练的内存优化技术已经取得了显著进展,从ZeRO优化器到Flash Attention等创新技术,为训练超大规模模型提供了可能。
|
4月前
|
缓存 固态存储 Windows
如何让内存发挥到最大效能?全面优化指南,提升电脑运行体验
电脑内存使用不合理会导致卡顿,本文教你如何优化内存性能。检查内存容量与主板支持上限,考虑升级或调整配置;关闭后台程序、管理浏览器标签、结束异常进程以释放内存;设置虚拟内存、调整视觉效果、定期重启提升效率;必要时增加内存条、选择高频内存、更换固态硬盘。避免盲目清理内存和依赖大内存忽视其他硬件瓶颈。只需合理设置,无需额外花钱,就能显著提升电脑速度。
|
4月前
|
存储 人工智能 自然语言处理
AI代理内存消耗过大?9种优化策略对比分析
在AI代理系统中,多代理协作虽能提升整体准确性,但真正决定性能的关键因素之一是**内存管理**。随着对话深度和长度的增加,内存消耗呈指数级增长,主要源于历史上下文、工具调用记录、数据库查询结果等组件的持续积累。本文深入探讨了从基础到高级的九种内存优化技术,涵盖顺序存储、滑动窗口、摘要型内存、基于检索的系统、内存增强变换器、分层优化、图形化记忆网络、压缩整合策略以及类操作系统内存管理。通过统一框架下的代码实现与性能评估,分析了每种技术的适用场景与局限性,为构建高效、可扩展的AI代理系统提供了系统性的优化路径和技术参考。
249 4
AI代理内存消耗过大?9种优化策略对比分析
|
7月前
|
缓存 并行计算 PyTorch
PyTorch CUDA内存管理优化:深度理解GPU资源分配与缓存机制
本文深入探讨了PyTorch中GPU内存管理的核心机制,特别是CUDA缓存分配器的作用与优化策略。文章分析了常见的“CUDA out of memory”问题及其成因,并通过实际案例(如Llama 1B模型训练)展示了内存分配模式。PyTorch的缓存分配器通过内存池化、延迟释放和碎片化优化等技术,显著提升了内存使用效率,减少了系统调用开销。此外,文章还介绍了高级优化方法,包括混合精度训练、梯度检查点技术及自定义内存分配器配置。这些策略有助于开发者在有限硬件资源下实现更高性能的深度学习模型训练与推理。
1440 0
|
4月前
|
存储 人工智能 API
AI代理性能提升实战:LangChain+LangGraph内存管理与上下文优化完整指南
在AI代理系统开发中,上下文工程成为提升系统性能的关键技术。本文探讨了从提示工程到上下文工程的转变,强调其通过为AI系统提供背景信息和工具支持,显著提升智能化程度和实用价值。文章系统分析了上下文工程的理论基础、核心策略(如写入、选择、压缩和隔离),并结合LangChain和LangGraph工具,展示了如何实现上下文工程技术以优化AI代理性能。通过Scratchpad机制、内存管理、RAG系统集成、多代理架构及沙盒环境等技术手段,开发者可以更高效地构建高性能、可扩展的AI系统。
499 0
AI代理性能提升实战:LangChain+LangGraph内存管理与上下文优化完整指南
|
5月前
|
缓存 监控 Cloud Native
Java Solon v3.2.0 高并发与低内存实战指南之解决方案优化
本文深入解析了Java Solon v3.2.0框架的实战应用,聚焦高并发与低内存消耗场景。通过响应式编程、云原生支持、内存优化等特性,结合API网关、数据库操作及分布式缓存实例,展示其在秒杀系统中的性能优势。文章还提供了Docker部署、监控方案及实际效果数据,助力开发者构建高效稳定的应用系统。代码示例详尽,适合希望提升系统性能的Java开发者参考。
268 4
Java Solon v3.2.0 高并发与低内存实战指南之解决方案优化
|
3月前
|
边缘计算 算法 Java
Java 绿色计算与性能优化:从内存管理到能耗降低的全方位优化策略与实践技巧
本文探讨了Java绿色计算与性能优化的技术方案和应用实例。文章从JVM调优(包括垃圾回收器选择、内存管理和并发优化)、代码优化(数据结构选择、对象创建和I/O操作优化)等方面提出优化策略,并结合电商平台、社交平台和智能工厂的实际案例,展示了通过Java新特性提升性能、降低能耗的显著效果。最终指出,综合运用这些优化方法不仅能提高系统性能,还能实现绿色计算目标,为企业节省成本并符合环保要求。
149 0
|
5月前
|
存储 自然语言处理 算法
基于内存高效算法的 LLM Token 优化:一个有效降低 API 成本的技术方案
本文探讨了在构建对话系统时如何通过一种内存高效算法降低大语言模型(LLM)的Token消耗和运营成本。传统方法中,随着对话深度增加,Token消耗呈指数级增长,导致成本上升。
466 7
基于内存高效算法的 LLM Token 优化:一个有效降低 API 成本的技术方案
|
6月前
|
缓存 编解码 Android开发
Android内存优化之图片优化
本文主要探讨Android开发中的图片优化问题,包括图片优化的重要性、OOM错误的成因及解决方法、Android支持的图片格式及其特点。同时介绍了图片储存优化的三种方式:尺寸优化、质量压缩和内存重用,并详细讲解了相关的实现方法与属性。此外,还分析了图片加载优化策略,如异步加载、缓存机制、懒加载等,并结合多级缓存流程提升性能。最后对比了几大主流图片加载框架(Universal ImageLoader、Picasso、Glide、Fresco)的特点与适用场景,重点推荐Fresco在处理大图、动图时的优异表现。这些内容为开发者提供了全面的图片优化解决方案。
270 1

热门文章

最新文章