You need to enable JavaScript to run this app.
最新活动
大模型
产品
解决方案
定价
生态与合作
支持与服务
开发者
了解我们

C语言malloc内存赋值后字节存储结果不符预期的疑问求助

C语言malloc内存赋值后字节存储结果不符预期的疑问求助

别担心,这个问题一点都不基础——刚接触字节、位操作的时候很容易犯这种想当然的错误,我当初也踩过类似的坑😉

核心问题拆解

你之所以没看到预期的“溢出截断”,本质是对位和字节的对应关系出现了认知偏差:

  1. 你最初误以为1字节=4位,所以觉得0b1111(十进制15)是单字节的最大值,但实际上现代计算机里的标准是1字节=8位——这也是你自己在编辑里纠正的点,但我们可以再结合你的代码深入解释。
  2. 你赋值的0b11111是十进制31,二进制仅占5位,远小于1字节(8位)的容量上限:
    • char是无符号类型(unsigned char),单字节最大值是0b11111111(十进制255)
    • char是有符号类型(默认大多是这个),单字节最大值是0b01111111(十进制127)
  3. 你的代码里malloc(4 * sizeof(char))分配了4个char的空间,每个char有8位,完全能容纳31这个值,自然不会发生截断,所以输出全是31。

如何触发你预期的“溢出截断”效果

如果想看到值被截断的现象,你可以尝试赋值超过8位的二进制数,比如0b111111111(9位,十进制511),这时候编译器会自动截断为低8位的0b11111111,对应的值会根据char的符号属性变化:

  • 若为signed char,最高位是符号位,0b11111111会被解释为-1
  • 若为unsigned char,则是十进制255

可以用这段代码验证:

#include <stdio.h>
#include <stdlib.h>

int main() {
    size_t n_bytes = 4;
    char *mmry_ptr = malloc(n_bytes * sizeof(char));
    if (mmry_ptr == NULL) { // 务必检查malloc是否成功!
        perror("malloc failed");
        return 1;
    }

    // 赋值9位的数,触发截断
    for (int i=0; i<(int)n_bytes; i++) {
        *(mmry_ptr+i) = 0b111111111; // 十进制511
    }

    // 同时打印有符号和无符号的结果,更直观
    for (int i=0; i<(int)n_bytes; i++) {
        printf("i=%d, addr=%p, signed_val=%d, unsigned_val=%u\n", 
               i, mmry_ptr+i, *(mmry_ptr+i), (unsigned char)*(mmry_ptr+i));
    }

    free(mmry_ptr); // 别忘了释放malloc的内存!
    return 0;
}

额外小提醒

  1. 永远记得检查malloc的返回值,避免空指针操作
  2. 用完malloc分配的内存后一定要用free释放,避免内存泄漏
  3. 如果需要精确控制位宽,建议用stdint.h里的固定位宽类型,比如uint8_t(无符号8位)、int8_t(有符号8位),避免不同编译器对char位宽的潜在差异(虽然绝大多数情况下都是8位,但严谨性很重要)

能主动写代码验证自己的理解,已经是非常好的学习习惯啦,继续加油!🤝

火山引擎 最新活动