当前位置: 首页 > news >正文

【C语言】memset(含常见用途、注意事项)

这里写目录标题

  • 1. memset用法及示例
  • 2.常见用途
    • 2.1 初始化数组
    • 2.2 清零动态分配的内存
    • 2.3 初始化结构体
    • 2.4 填充非零值
  • 3.注意事项
  • 4. 总结

1. memset用法及示例

memset 是 C 标准库中的一个函数,用于将一段内存区域填充为指定的值。它通常用于初始化数组、结构体或动态分配的内存。

#include <string.h>  // 需要包含头文件
void *memset(void *ptr, int value, size_t num);

参数:

  • ptr:指向要填充的内存区域的指针。
  • value:要填充的值(以 int 形式传递,但实际填充时会转换为 unsigned char)。
  • num:要填充的字节数。

返回值:返回指向 ptr 的指针。

2.常见用途

  • 初始化数组:将数组的所有元素设置为特定值。
  • 清零内存:将内存区域设置为 0。
  • 初始化结构体:将结构体的内存区域设置为特定值。

示例代码

2.1 初始化数组

将数组的所有元素设置为 0:

#include <stdio.h>
#include <string.h>

int main() {
    int arr[10];
    memset(arr, 0, sizeof(arr));  // 将数组所有元素设置为 0

    for (int i = 0; i < 10; i++) {
        printf("arr[%d] = %d\n", i, arr[i]);
    }

    return 0;
}

输出:

arr[0] = 0
arr[1] = 0
...
arr[9] = 0

2.2 清零动态分配的内存

将动态分配的内存设置为 0:

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

int main() {
    int *ptr = (int *)malloc(10 * sizeof(int));
    if (ptr == NULL) {
        printf("Memory allocation failed\n");
        return 1;
    }

    memset(ptr, 0, 10 * sizeof(int));  // 将动态分配的内存清零

    for (int i = 0; i < 10; i++) {
        printf("ptr[%d] = %d\n", i, ptr[i]);
    }

    free(ptr);  // 释放内存
    return 0;
}

输出:

ptr[0] = 0
ptr[1] = 0
...
ptr[9] = 0

2.3 初始化结构体

将结构体的内存区域设置为特定值:

#include <stdio.h>
#include <string.h>

struct Point {
    int x;
    int y;
};

int main() {
    struct Point p;
    memset(&p, 0, sizeof(p));  // 将结构体清零

    printf("p.x = %d, p.y = %d\n", p.x, p.y);

    return 0;
}

输出:

p.x = 0, p.y = 0

2.4 填充非零值

将数组的所有字节设置为 1:

int main() {
    int arr[10];
    memset(arr, 1, sizeof(arr));  // 将数组的每个字节设置为 1
    printf("%d\n",sizeof(arr));
    for (int i = 0; i < 10; i++) {
        printf("arr[%d] = %08x\n", i, arr[i]);  // 打印十六进制值
    }

    return 0;
}

输出:

40
arr[0] = 01010101
arr[1] = 01010101
...
arr[9] = 01010101

注意:memset 是按字节填充的,因此 int 类型的每个字节都会被设置为 1,而不是整个 int 值变为 1。

如果是想将数组中的每个 元素 设置为 1,而不是将每个 字节 设置为 1,不能直接使用 memset,因为 memset 是按字节操作的。 对于 int 类型的数组,每个元素占 4 个字节(假设 int 是 32 位),memset 会将每个字节设置为 1,导致每个 int 元素的值实际上是 0x01010101(即十进制的 16843009)。

正确方法
要将数组中的每个元素设置为 1,可以使用循环逐个赋值:

  1. 使用循环初始化
#include <stdio.h>

int main() {
    int arr[10];

    // 将数组的每个元素设置为 1
    for (int i = 0; i < 10; i++) {
        arr[i] = 1;
    }

    // 打印数组内容
    for (int i = 0; i < 10; i++) {
        printf("arr[%d] = %08x\n", i, arr[i]);
    }

    return 0;
}

输出:

arr[0] = 00000001
arr[1] = 00000001
...
arr[9] = 00000001
  1. 使用 for 循环的简洁写法 C99 及以上版本支持在循环中直接定义变量:
for (int i = 0; i < 10; i++) {
    arr[i] = 1;
}
  1. 使用 memset 的替代方案
    如果使用类似 memset 的方式,可以编写一个宏或函数来实现:
#include <stdio.h>
#include <string.h>

#define SET_ARRAY(arr, value, size) \
    for (size_t i = 0; i < size; i++) { \
        arr[i] = value; \
    }

int main() {
    int arr[10];
    SET_ARRAY(arr, 1, 10);  // 将数组的每个元素设置为 1

    // 打印数组内容
    for (int i = 0; i < 10; i++) {
        printf("arr[%d] = %08x\n", i, arr[i]);
    }

    return 0;
}

为什么不能用 memset?
memset 是按字节填充的,而 int 类型的每个元素占 4 个字节(32 位)。当你调用:
memset(arr, 1, sizeof(arr));
实际上是将每个字节设置为 1,因此每个 int 元素的值是:0x01010101 即十进制的 16843009,而不是 1。

总结
方法 适用场景 代码示例
循环赋值 初始化数组元素为特定值 for (int i = 0; i < size; i++) arr[i] = 1;
宏定义 简化代码,支持任意类型 #define SET_ARRAY(arr, value, size) …
memset 按字节填充(不适合初始化 int) memset(arr, 0, sizeof(arr))
如果你需要将数组的每个元素设置为 1,推荐使用循环赋值的方式。

3.注意事项

按字节填充:

memset 是按字节填充的,因此对于非字符类型的数组(如 int 数组),填充的值可能不符合预期。

例如,memset(arr, 1, sizeof(arr)) 会将每个字节设置为 1,而不是将每个 int 元素设置为 1。

清零内存:

使用 memset(ptr, 0, num) 可以安全地将内存区域清零。

性能:

memset 通常由编译器优化为高效的机器指令,适合大规模内存操作。

4. 总结

场景 示例代码
初始化数组 memset(arr, 0, sizeof(arr))
清零动态内存 memset(ptr, 0, size * sizeof(int))
初始化结构体 memset(&struct, 0, sizeof(struct))
填充非零值 memset(arr, 1, sizeof(arr))
通过 memset,可以快速初始化或填充内存区域,但需注意其按字节填充的特性

相关文章:

  • AWS Bedrock 正式接入 DeepSeek-R1 模型:安全托管的生成式 AI 解决方案
  • 2025年【A特种设备相关管理】免费试题及A特种设备相关管理模拟考试题库
  • 哪些业务场景更适合用MongoDB?何时比MySQL/PostgreSQL好用?
  • JavaScript性能优化
  • HTML基础
  • 深度学习实验
  • upload-labs-master通关攻略(1~4)
  • Linux内核如何和设备树协同工作的?
  • Oracle中Unique(id)和id int unique的区别
  • Redis常用数据结构及命令详解:从基础到进阶
  • Qt Creator插件系统详解及插件开发实战
  • 【后端】【ubuntu】 ubuntu目录权限查看的几种方法
  • 《旅游与摄影》是什么级别的期刊?是正规期刊吗?能评职称吗?
  • 【脚本】Linux一键扩大虚拟内存的大小
  • 多光谱相机数据采集过程中常见仪器
  • 基于FPGA的图像退化算法verilog实现,分别实现横向和纵向运动模糊,包括tb和MATLAB辅助验证
  • ARM SVC指令
  • FX-友元函数和友元类
  • C++学习——顺序表(二)
  • CSS-三大特性,盒子模型,圆角边框,盒子阴影,文字阴影
  • 国家统计局:4月份居民消费价格同比下降0.1%
  • 秦洪看盘|交易型资金收缩,释放短线压力
  • 北外滩集团21.6亿元摘上海虹口地块,为《酱园弄》取景地
  • 101条关于减重的知识,其中一定有你不知道的
  • 英国和美国就关税贸易协议条款达成一致
  • Meta正为AI眼镜开发人脸识别功能