将 memset 与未初始化的变量一起使用

Posted

技术标签:

【中文标题】将 memset 与未初始化的变量一起使用【英文标题】:Using memset with uninitialized variables 【发布时间】:2021-05-10 19:47:12 【问题描述】:

这是没有未定义行为的有效 C 代码吗?

int main()
 int a;
 memset(&a, 5, sizeof(int));

 return a;

我假设这等于只是做int a = 5

我试图了解在上面的示例中仅声明一个变量(不定义它)是否足以将其放入堆栈。

【问题讨论】:

C 没有堆栈的概念。只有变量,堆栈是实现细节。 这不是声明。这是一个定义。只是它的初始值是不确定的。 定义在做int a = 5 @EugeneSh.:C 中的每个定义都是一个声明。没有什么可以是定义,也不能是声明。 @AdrianMole:它在 C 2018 6.7 中。首先,它是一个声明说明符列表,后跟一个带有可选初始化的声明符列表,然后是一个分号。 【参考方案1】:

这是没有未定义行为的有效 C 代码吗?

是的——一旦a变量在给定范围内声明(如函数或其他 ... 分隔块),获取其地址并使用该地址访问变量是有效的在该范围内范围(就像您的memset 电话一样)。 当该范围结束时(即不再是“活动”)尝试使用该地址将导致未定义的行为;比如下面是UB:

int main()

    int* p;
     // New scope ...
        int a;
        p = &a; // Pointer to "a" is valid HERE
     // The scope of "a" (and its 'lifetime') ends here
    memset(p, 5, sizeof(int)); // INVALID: "p" now points to a dead (invalid) variable

但是,您的代码示例中有一个主要警告……

我假设这等于只执行 int a = 5。

问题是:它将5 分配给a 变量的每个组件字节,所以它这样做(假设是4 字节int):

int a = 0x05050505;

等同于:

int a = 84215045;

【讨论】:

抱歉改成char @Dan 不要改变你的问题。它使已经提供的答案无效。 所以可以肯定地说,只在这里声明一个变量把它放在堆栈上,因此得到它的地址是有效的? @Dan 是:一旦在给定范围(函数或其他 ... 分隔块)中声明了变量,获取其地址(并主要使用它)是有效的。 @AdrianMole 谢谢,你能对此发表答案吗?我会接受的,其他所有答案都集中在我遇到的数据类型错误上。【参考方案2】:

来自 C 标准(7.23.6.1 的 memset 函数)

2 memset 函数复制 c 的值(转换为无符号 char) 到指向的对象的前 n 个字符中的每一个 秒。

所以这个电话

memset(&a, 5, sizeof(int));

不将变量a 设置为等于5。在内部,变量看起来像

0x05050505

这是一个演示程序

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

int main(void) 

    int a;
    
    memset( &a, 5, sizeof( int ) );
    
    printf( "%#x\n", ( unsigned )a );
    
    return 0;

它的输出是

0x5050505

您应该谨慎使用带有整数的函数memset,因为通常它会产生一个陷阱值。此外,结果取决于内部整数如何从 MSB 或 LSB 开始存储。

附:您在没有链接的块范围内声明了一个变量。它也是一个具有自动存储持续时间的变量定义。由于变量未显式初始化,因此它具有不确定的值。您可以应用运算符&amp; 的地址来获取定义变量的内存范围的地址。

【讨论】:

抱歉改成char 好的,可以这么说吗,只在这里声明一个变量把它放在堆栈上,因此获取它的地址是有效的? @Barmar 这是一个错字。:) @Dan 为什么获取地址无效?您声明(并定义)了一个对象。 我不知道,我以为只声明它可能会将其地址传递给 memset UB,C 很奇怪。【参考方案3】:

这不是未定义的行为。问题是它不是你所期望的。

结果

memset(&a, 5, sizeof(int));

包括将整数a 的四个字节中的每一个设置为5

【讨论】:

以上是关于将 memset 与未初始化的变量一起使用的主要内容,如果未能解决你的问题,请参考以下文章

memset与初始化

为啥memset不能将数组元素初始化为1

memset结构体初始化

memset使用技巧

memset用法总结

memset中需要的一些注意问题