assignment operator vs. copy constructor C++

前端 未结 3 1284
我在风中等你
我在风中等你 2021-02-01 05:34

I have the following code to test out my understanding of basic pointers in C++:

// Integer.cpp
#include \"Integer.h\"
Integer::Integer()
{
  value = new int;
           


        
相关标签:
3条回答
  • 2021-02-01 06:31

    The problem you're experiencing is caused by the default copy constructor, which copies the pointer but doesn't associate it with newly allocated memory (like your implementation of copy constructor does). When you pass object by value, a copy is created and when the execution goes out of scope, this copy is destructed. delete from the destructor invalidates the value pointer of intVal1 object, making it dangling pointer, dereferencing of which causes undefined behavior.

    Debug outputs might be used to understand the behavior of your code:

    class Integer {
    public:
    
        Integer() {
          cout << "ctor" << endl;
          value = new int;
          *value = 0;
        }
    
        ~Integer() {
            cout << "destructor" << endl;
            delete value;
        }
    
        Integer(int intVal) {
          cout << "ctor(int)" << endl;
          value = new int;
          *value = intVal;
        } 
    
        Integer(const Integer &rhInt) {
          cout << "copy ctor" << endl;
          value = new int;
          *value = *rhInt.value;
        }
    
        Integer& operator=(const Integer& rhInt){   
          cout << "assignment" << endl;
          *value = *rhInt.value;
          return *this;
        }
    
        int *value;
    };
    
    void foo(Integer intObj) {
        cout << intObj.value << " " << *(intObj.value) << endl;
    }
    

    Now output of this code:

    Integer intVal1;
    Integer intVal2(10);
    
    foo( intVal1 );
    foo( intVal2 );
    
    intVal1 = intVal2;
    
    foo( intVal1 );
    

    is:

    ctor
    ctor(int)
    copy ctor
    0x9ed4028 0
    destructor
    copy ctor
    0x9ed4038 10
    destructor
    assignment
    copy ctor
    0x9ed4048 10
    destructor
    destructor
    destructor

    which shows that copy constructor is used when passing objects by value. However, important to notice here is the destructor called upon the return from your function. And if you remove your implementation of copy constructor, then the output is:

    ctor
    ctor(int)
    0x8134008 0
    destructor
    0x8134018 10
    destructor
    assignment
    0x8134008 135479296
    destructor
    destructor
    destructor

    showing that the first copy called delete on the same pointer (pointing to 0x8134008) as was used by third copy later, where the memory pointed by this dangling pointer has been used.

    0 讨论(0)
  • 2021-02-01 06:34

    Think about this call:

    displayInteger( "intVal1", intVal1 );
    

    You are creating a copy of intVal1 into the intObj parameter of displayInteger:

    void displayInteger( char* str, Integer intObj )
    {
      cout << str << " is " << intObj.getInteger() << endl;
    }
    

    That copy will be pointing to the same int that intVal1 is. When displayInteger returns, intObj is destroyed, which will cause the int to be destroyed, and the pointer in intVal1 to be pointing to an invalid object. At that point all bets are off (A.K.A. undefined behavior) if you try to access the value. A similar thing happens for intVal2.

    At a more general level, by removing the copy constructor, you are violating the Rule of Three, which typically leads to these kinds of problems.

    0 讨论(0)
  • 2021-02-01 06:37

    The copy constructor is not used during assignment. Copy constructor in your case is used when passing arguments to displayInteger function. The second parameter is passed by value, meaning that it is initailized by copy contructor.

    Your version of copy constructor performs deep copying of data owned by the class (just like your assignment operator does). So, everything works correctly with your version of copy constructor.

    If you remove your own copy constructor, the compiler will generate one for you implicitly. The compiler-generated copy constructor will perform shallow copying of the object. This will violate the "Rule of Three" and destroy the functionality of your class, which is exactly what you observe in your experiment. Basically, the first call to displayInteger damages your intVal1 object and the second call to displayInteger damages your intVal2 object. After that both of your objects are broken, which is why the third displayInteger call displays garbage.

    If you change the declaration of displayInteger to

    void displayInteger( char* str, const Integer &intObj )
    

    your code will "work" even without an explicit copy constructor. But it is not a good idea to ignore the "Rule of Three" in any case. A class implemented in this way either has to obey the "Rule of Three" or has to be made non-copyable.

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