- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
需要说明的是:我确实知道 malloc
和 free
是在 C 库中实现的,它通常从操作系统分配内存块,并进行自己的管理以将较小的内存分配给应用程序并跟踪分配的字节数。这个问题不是How does free know how much to free .
相反,我想知道为什么free
一开始就是这样制作的。作为一门低级语言,我认为让 C 程序员不仅跟踪分配了多少内存,而且跟踪分配了多少内存是完全合理的(事实上,我通常发现我最终会跟踪字节数无论如何都被分配了)。我还发现明确地将字节数提供给 free
可能允许一些性能优化,例如具有用于不同分配大小的单独池的分配器将能够仅通过查看输入参数来确定要从哪个池中释放,并且总体上空间开销会更少。
所以,简而言之,为什么 malloc
和 free
创建以便他们需要在内部跟踪分配的字节数?这只是历史事故吗?
一个小编辑:
一些人提供了诸如“如果您释放的金额与分配的金额不同会怎样”之类的观点。我想象的 API 可能只需要一个来释放分配的字节数;释放或多或少可以简单地由 UB 或实现定义。不过,我不想阻止关于其他可能性的讨论。
最佳答案
单参数 free(void *)
(在 Unix V7 中引入)与之前的双参数 mfree(void *, size_t)
相比有另一个主要优势,我在这里没有看到它:一个参数 free
极大地简化了所有其他与堆内存一起使用的 API。例如,如果 free
需要内存块的大小,那么 strdup
将不得不以某种方式返回两个值(指针 + 大小)而不是一个(指针),并且 C 使多值返回比单值返回要麻烦得多。我们必须写 char *strdup(char *)
或 char *strdup(char *, size_t *)
而不是 struct CharPWithSize { char *val; size_t size}; CharPWithSize strdup(char *)
。 (现在,第二个选项看起来很诱人,因为我们知道以 NUL 结尾的字符串是 "most catastrophic design bug in the history of computing" ,但这是事后的看法。早在 70 年代,C 将字符串作为简单的 char *
处理的能力实际上被认为是 defining advantage over competitors like Pascal and Algol,它。受此问题影响的不仅仅是 strdup
- 它影响分配堆内存的每个系统或用户定义的函数。
早期的 Unix 设计者是非常聪明的人,free
比 mfree
更好的原因有很多,所以基本上我认为问题的答案是他们注意到了这一点并相应地设计了他们的系统。我怀疑你会在他们做出这个决定的那一刻找到他们头脑中发生的事情的任何直接记录。但我们可以想象。
假设您正在用 C 编写应用程序以在 V6 Unix 上运行,其两个参数为 mfree
。到目前为止,您已经管理得很好,但是随着您的程序 become more ambitious 并且需要越来越多地使用堆分配的变量,跟踪这些指针大小变得越来越麻烦。但是你有一个绝妙的主意:而不是一直复制这些 size_t
,你可以编写一些实用函数,将大小直接存储在分配的内存中:
void *my_alloc(size_t size) {
void *block = malloc(sizeof(size) + size);
*(size_t *)block = size;
return (void *) ((size_t *)block + 1);
}
void my_free(void *block) {
block = (size_t *)block - 1;
mfree(block, *(size_t *)block);
}
你使用这些新函数编写的代码越多,它们看起来就越棒。它们不仅使您的代码更易于编写,而且还使您的代码更快——这两件事通常不会同时出现!在你到处传递这些
size_t
之前,这增加了复制的 CPU 开销,意味着你不得不更频繁地溢出寄存器(特别是对于额外的函数参数),并浪费内存(因为嵌套函数调用经常导致
size_t
的多个副本存储在不同的堆栈帧中)。在你的新系统中,你仍然需要花费内存来存储
size_t
,但只有一次,而且它永远不会被复制到任何地方。这些可能看起来效率很低,但请记住,我们谈论的是具有 256 KiB RAM 的高端机器。
strdup
,他们意识到使用你很酷的技巧的人将无法使用他们的新函数,因为他们的新函数都使用笨重的指针+大小 API。然后让你伤心过,因为你意识到你必须自己重写好
strdup(char *)
功能在每次你写的程序,而不是能够使用的系统版本。
string.h
和
malloc
是供应商扩展 (!)。因此,建议 Bearded Man #1,我们可以随心所欲地改变它们;为什么我们不直接将您棘手的分配器宣布为官方分配器?
strdup
,人们会实际使用它,而 Bearded Man #2 —— 相信他已经赢得了一点时间- 回到
messing around with quines 。发货吧!
关于c - 为什么 C 中的 `free` 不占用要释放的字节数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24203940/
我正在阅读 SQL/92(我是新手),它经历了不同的数据类型。其中之一是CHAR,我当然知道它与java中的String非常相似,而不是java中的char。但我们假设它是 CHAR(1)。只有一个字
我的 mysqld 进程消耗了 232% 的 CPU,并且有 14000 多个连接 (我对这件事有点陌生,但关注 Stack Overflow 寻求帮助)。 顶部: PID USER P
Tomcat 服务器占用 100% 的 CPU,但仅在 PRD。我们无法在其他环境中重现这一点。 进行线程转储后,我们发现有一些线程处于等待/可运行状态,但无法找到我们如何找到根本原因。 你能帮忙吗?
我正在使用 Xcode、SpriteKit 和 Swift 构建我的第一款 iPhone 游戏。我对这些技术不熟悉,但我熟悉一般的编程概念。 这是我想用英语做的事情。我想让圆圈随机出现在屏幕上,然后开
我的套接字消耗了 100% 的计算机 CPU。有 150 个客户端每 30 秒异步向服务器发送消息。有谁知道如何解决这个问题?下面是我的 ServerSocket 类 public class Ser
一段时间后(有时几分钟,有时几天),我的应用开始消耗 100% 的 CPU。正如我从 VisualVM 看到的那样,它总是发生在 org.elasticsearch.common.netty.chan
在我的容器 Controller 中,用户可以平移 View 以切换到不同的 View 。当平移手势开始时,它会将新 View Controller 的 View 添加到 View 中:view.in
假设我在数据框中有两列,其中一列不完整。 df = pd.DataFrame({'a': [1, 2, 3, 4], 'b':[5, '', 6, '']}) df Out: a b
在Ubuntu 16.04 LTS中,pyteserract脚本吃得太高,导致系统间歇性重启。 top命令输出为 top - 21:23:31 up 27 min, 4 users, lo
我在具有 88 个内核和 60 个 reducer 的 hadoop 集群上运行 mapreduce 作业。由于某种原因,它只使用了 79 个集群核心。开始时它运行 79 个映射器,但当完成一半拆分时
我正在对机器上的所有用户进行查询,当它执行时,它会占用 100% 的 CPU 并锁定系统。我已经等了 5 分钟,但什么也没有发生。 在任务管理器中,wmiprvse.exe 占用了所有 CPU。当我终
我正在从套接字(通过 TCP 协议(protocol))读取消息,但我注意到 CPU 花费大量时间来调用 BufferedInputStream 的 available() 方法。这是我的代码:
我有 6 个线程。其中一个线程进入某个范围并打开“锁定”和所有其他线程线程正在等待并希望进入相同的范围。 现在,其他线程是否会获得 CPU 时间?其他线程是否在线程调度中?我知道所有其他线程都处于等待
我正在尝试创建一个社交媒体应用程序。但它需要大约 300mb 内存。所以我的主页上有 5 个包含帖子的 fragment 。总体内存使用量为 250-300mb 然后为了测试,我禁用了这些 fragm
我有一个带有一些 TextFormField 的表单,我想扩展最后一个 TextFormField 以占据屏幕的其余部分。最后一个 TextFormField 可以有多行文本。 我没能做到这一点,并尝
我收到磁盘几乎已满的警告,所以我运行 DaisyDisk .. 显然 Xcode 占用了 15GB 的空间: http://imgur.com/a/cTIZZ iOS 设备支持为 12.3 GB: h
我正在使用 Xcode Playground 研究 Swift 内存布局,我创建了一个带有 bool、double 和 int32 的结构,如下所示。基于这种结构,MemoryLayout 的打印结果
一旦执行“self.navigationController pushviewcontroller:vc animated:YES”,我的 CPU 就会达到 100%。我在 Stack Overflo
警告:CPU 使用率达到 100%,请小心。 Link to the jsFiddle 编写此脚本是为了设计动态蛇梯板。每次刷新页面时,都会创建一个新板。大多数时候所有的背景图像都不会出现,CPU 使
我不知道为什么,但是MYSQL给CPU带来了很大的负载。我必须每秒多次更新数据库,并且用户群正在不断增长。 一开始还好,但是现在 CPU 负载每天都在增加 这是日志中的慢速查询: *Query_tim
我是一名优秀的程序员,十分优秀!