Undefined behavior from pointer math on a C++ array

前端 未结 3 1143
逝去的感伤
逝去的感伤 2021-02-05 06:36

Why the output of this program is 4?

#include 

int main()
{
    short A[] = {1, 2, 3, 4, 5, 6};
    std::cout << *(short*)((c         


        
3条回答
  •  北海茫月
    2021-02-05 07:20

    You are violating strict aliasing rules here. You can't just read half-way into an object and pretend it's an object all on its own. You can't invent hypothetical objects using byte offsets like this. GCC is perfectly within its rights to do crazy sh!t like going back in time and murdering Elvis Presley, when you hand it your program.

    What you are allowed to do is inspect and manipulate the bytes that make up an arbitrary object, using a char*. Using that privilege:

    #include 
    #include 
    
    int main()
    {
        short A[] = {1, 2, 3, 4, 5, 6};
    
        short B;
        std::copy(
           (char*)A + 7,
           (char*)A + 7 + sizeof(short),
           (char*)&B
        );
        std::cout << std::showbase << std::hex << B << std::endl;
    }
    
    // Output: 0x500
    

    (live demo)

    But you can't just "make up" a non-existent object in the original collection.

    Furthermore, even if you have a compiler that can be told to ignore this problem (e.g. with GCC's -fno-strict-aliasing switch), the made-up object is not correctly aligned for any current mainstream architecture. A short cannot legally live at that odd-numbered location in memory, so you doubly can't pretend there is one there. There's just no way to get around how undefined the original code's behaviour is; in fact, if you pass GCC the -fsanitize=undefined switch it will tell you as much.

    I'm simplifying a little.

提交回复
热议问题