RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1347894
Accepted
Escator
Escator
Asked:2022-04-10 01:24:51 +0000 UTC2022-04-10 01:24:51 +0000 UTC 2022-04-10 01:24:51 +0000 UTC

如何避免内存泄漏(被调用函数中的动态分配)

  • 772

纯 C (C 11) 中有一个任务,用原型编写函数: void *to_upper(const char *str) 应该返回指向字符串 str 副本的指针,其中所有字符都转换为大写。我是这样实现的:

void *to_upper(const char *str) {
    void *res = NULL;
    ssize_t len = strlen(str) + 1;
    char *str1 = (char*) malloc(sizeof(char) * (len + 1));
    for (size_t i = 0; i < len; i++) {
        if (*(str + i) > 96 && *(str + i) < 123)
            *(str1 + i) = *(str + i) - 32;
        else
            *(str1 + i) = *(str + i);
        res = (void*) str1;
    }
    *(str1 + len) = '\0';
    return res;
}

但我知道,通过为新行动态分配内存而不是释放它,每次使用此函数时都会发生泄漏。有没有办法在没有内存泄漏的情况下实现这个功能?

c
  • 2 2 个回答
  • 10 Views

2 个回答

  • Voted
  1. gbg
    2022-04-10T01:42:15Z2022-04-10T01:42:15Z

    不严肃的回答:

    在上面添加评论:

    //Кто забудет вызвать free() для полученного 
    //отсюда указателя будет уволен на месте!
    

    更严肃的回答:

    在古老的《DOOM》中,一种称为区域记忆的机制负责这一点。当时的想法是,内存是从公共池中动态分配的,然后,当游戏在关卡之间转换时,肯定不需要数据时,内存就被一次性清除了。

    您也可以这样做 - 在应用程序中选择一个时刻,当所有选定的行肯定都失去了相关性时 - 并一次将它们全部拆除。

    • 1
  2. Best Answer
    KoVadim
    2022-04-10T02:48:14Z2022-04-10T02:48:14Z

    没有人会费心制作一个静态缓冲区并返回一个指向它的指针。的确,退出函数后,最好立即将其复制给自己。此外,该技术用于系统库中的某些功能。我已经以与问题中的代码相同的样式绘制了一个示例

    void *to_upper(const char *str) {
        #define MLEN 4096
        static char buffer[MLEN];
        ssize_t len = strlen(str) + 1;
        if (len >= MLEN) return NULL;
        for (size_t i = 0; i < len; i++) {
            if (str[i] >= 'a' && str[i] <= 'z')
                buffer[i] = str[i] - 32;
            else
                buffer[i] = str[i];
        }
        buffer[len] = '\0';
        return &buffer;
    }
    

    在这里,和往常一样,4kb 对每个人来说都足够了 :) 我还用真实字符替换了魔法常数。

    此外,这种技术通常用于“加载”代码中,不需要额外的内存分配。

    这样的函数根本不能同时在两个线程中使用。但这可以解决。

    - - 更新 - -

    但如果这是一次采访,@avp 在那里问:“我想要这样的代码

    int main()
    {
        printf("%s %s\n", (char *)to_upper("data"), (char *)to_upper("foo"));
    }
    

    工作正常。好吧,至少它几乎总是正确的。”

    让我们回答这个代码。

    void *to_upper(const char *str) {
        #define MLEN 4096
        #define SIZE 16
        static char buffer[SIZE][MLEN];
        static int pos = SIZE-1;
        pos = (pos + 1) % SIZE;
        ssize_t len = strlen(str) + 1;
        if (len >= MLEN) return NULL;
        for (size_t i = 0; i < len; i++) {
            if (str[i] >= 'a' && str[i] <= 'z')
                buffer[pos][i] = str[i] - 32;
            else
                buffer[pos][i] = str[i];
        }
        buffer[pos][len] = '\0';
        return &buffer[pos];
    }
    

    我们现在没有一个缓冲区,而是 16 个。每个新请求都由一个单独的缓冲区提供服务。是的,当它们绕圈子时,但在大多数情况下,即使是多线程也可以工作(如果您提供线程安全的增量和正确的缓冲区选择。但是这些调整是老先生们无事可做时的任务.

    • 1

相关问题

  • free 出于某种原因不会从内存中删除数组

  • 请帮助代码

  • 为什么 masm 对字符串或文本文字太长发誓,为什么在结构中设置 db 或 dw?

  • 如何将数字拆分为位并将其写入 C 中的数组?

  • 如何以给定的角度移动物体?

  • 解决“子集和问题”的时效算法

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    表格填充不起作用

    • 2 个回答
  • Marko Smith

    提示 50/50,有两个,其中一个是正确的

    • 1 个回答
  • Marko Smith

    在 PyQt5 中停止进程

    • 1 个回答
  • Marko Smith

    我的脚本不工作

    • 1 个回答
  • Marko Smith

    在文本文件中写入和读取列表

    • 2 个回答
  • Marko Smith

    如何像屏幕截图中那样并排排列这些块?

    • 1 个回答
  • Marko Smith

    确定文本文件中每一行的字符数

    • 2 个回答
  • Marko Smith

    将接口对象传递给 JAVA 构造函数

    • 1 个回答
  • Marko Smith

    正确更新数据库中的数据

    • 1 个回答
  • Marko Smith

    Python解析不是css

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5