zeroing out memory

后端 未结 12 2035
無奈伤痛
無奈伤痛 2021-01-31 09:52

gcc 4.4.4 C89

I am just wondering what most C programmers do when they want to zero out memory.

For example, I have a buffer of 1024 bytes. Sometimes I do this:<

相关标签:
12条回答
  • 2021-01-31 10:11

    The worst that can happen by not doing it is that you write some data in character by character and later interpret it as a string (and you didn't write a null terminator). Or you end up failing to realise a section of it was uninitialised and read it as though it were valid data. Basically: all sorts of nastiness.

    Memset should be fine (provided you correct the sizeof typo :-)). I prefer that to your first example because I think it's clearer.

    For dynamically allocated memory, I use calloc rather than malloc and memset.

    0 讨论(0)
  • 2021-01-31 10:14

    Depends how you're filling it: if you're planning on writing to it before even potentially reading anything, then why bother? It also depends what you're going to use the buffer for: if it's going to be treated as a string, then you just need to set the first byte to \0:

    char buffer[1024];
    buffer[0] = '\0';
    

    However, if you're using it as a byte stream, then the contents of the entire array are probably going to be relevant, so memseting the entire thing or setting it to { 0 } as in your example is a smart move.

    0 讨论(0)
  • 2021-01-31 10:16

    I also use memset(buffer, 0, sizeof(buffer));

    The risk of not using it is that there is no guarantee that the buffer you are using is completely empty, there might be garbage which may lead to unpredictable behavior.

    Always memset-ing to 0 after malloc, is a very good practice.

    0 讨论(0)
  • 2021-01-31 10:18

    I'm not familiar with the:

    char buffer[1024] = {0};
    

    technique. But assuming it does what I think it does, there's a (potential) difference to the two techniques.

    The first one is done at COMPILE time, and the buffer will be part of the static image of the executable, and thus be 0's when you load.

    The latter will be done at RUN TIME.

    The first may incur some load time behaviour. If you just have:

    char buffer[1024];
    

    the modern loaders may well "virtually" load that...that is, it won't take any real space in the file, it'll simply be an instruction to the loader to carve out a block when the program is loaded. I'm not comfortable enough with modern loaders say if that's true or not.

    But if you pre-initialize it, then that will certainly need to be loaded from the executable.

    Mind, neither of these have "real" performance impacts in the small. They may not have any in the "large". Just saying there's potential here, and the two techniques are in fact doing something quite different.

    0 讨论(0)
  • 2021-01-31 10:23

    In this particular case, there's not much difference. I prefer = { 0 } over memset because memset is more error-prone:

    • It provides an opportunity to get the bounds wrong.
    • It provides an opportunity to mix up the arguments to memset (e.g. memset(buf, sizeof buf, 0) instead of memset(buf, 0, sizeof buf).

    In general, = { 0 } is better for initializing structs too. It effectively initializes all members as if you had written = 0 to initialize each. This means that pointer members are guaranteed to be initialized to the null pointer (which might not be all-bits-zero, and all-bits-zero is what you'd get if you had used memset).

    On the other hand, = { 0 } can leave padding bits in a struct as garbage, so it might not be appropriate if you plan to use memcmp to compare them later.

    0 讨论(0)
  • 2021-01-31 10:25

    The worst that can happen? You end up (unwittingly) with a string that is not NULL terminated, or an integer that inherits whatever happened to be to the right of it after you printed to part of the buffer. Yet, unterminated strings can happen other ways, too, even if you initialized the buffer.

    Edit (from comments) The end of the world is also a remote possibility, depending on what you are doing.

    Either is undesirable. However, unless completely eschewing dynamically allocated memory, most statically allocated buffers are typically rather small, which makes memset() relatively cheap. In fact, much cheaper than most calls to calloc() for dynamic blocks, which tend to be bigger than ~2k.

    c99 contains language regarding default initialization values, I can't, however, seem to make gcc -std=c99 agree with that, using any kind of storage.

    Still, with a lot of older compilers (and compilers that aren't quite c99) still in use, I prefer to just use memset()

    0 讨论(0)
提交回复
热议问题