Why can I access private variables in the copy constructor?
Solution 1:
The access modifiers work on class level, and not on object level.
That is, two objects of the same class can access each others private data.
Why:
Primarily due to efficiency. It would be a non-negligible runtime overhead to check if this == other
each time you access other.x
which you would have to if the access modifiers worked on object level.
It's also kind of semantically logical if you think of it in terms of scoping: "How big part of the code do I need to keep in mind when modifying a private variable?" – You need to keep the code of the whole class in mind, and this is orthogonal to which objects exist in runtime.
And it's incredibly convenient when writing copy constructors and assignment operators.
Solution 2:
IMHO, existing answers do a poor job explaining the "Why" of this - focusing too much on reiterating what behaviour's valid. "access modifiers work on class level, and not on object level." - yes, but why?
The overarching concept here is that it's the programmer(s) designing, writing and maintaining a class who is(are) expected to understand the OO encapsulation desired and empowered to coordinate its implementation. So, if you're writing class X
, you're encoding not just how an individual X x
object can be used by code with access to it, but also how:
- derived classes are able to interact with it (through optionally-pure virtual functions and/or protected access), and
- distinct
X
objects cooperate to provide intended behaviours while honouring the post-conditions and invariants from your design.
It's not just the copy constructor either - a great many operations can involve two or more instances of your class: if you're comparing, adding/multiplying/dividing, copy-constructing, cloning, assigning etc. then it's often the case that you either simply must have access to private and/or protected data in the other object, or want it to allow a simpler, faster or generally better function implementation.
Specifically, these operations may want to take advantage of priviledged access to do things like:
- (copy constructors) use a private member of the "rhs" (right hand side) object in an initialiser list, so that a member variable is itself copy-constructed instead of default-constructed (if even legal) then assigned too (again, if legal)
- share resources - file handles, shared memory segments,
shared_ptr
s to reference data etc. - take ownership of things, e.g.
auto_ptr<>
"moves" ownership to the object under construction - copy private "cache", calibration, or state members needed to construct the new object in an optimally usable state without having to regenerate them from scratch
- copy/access diagnostic/trace information kept in the object being copied that's not otherwise accessible through public APIs but might be used by some later exception object or logging (e.g. something about the time/circumstances when the "original" non-copy-constructed instance was constructed)
- perform a more efficient copy of some data: e.g. objects may have e.g. an
unordered_map
member but publicly only exposebegin()
andend()
iterators - with direct access tosize()
you couldreserve
capacity for faster copying; worse still if they only exposeat()
andinsert()
and otherwisethrow
.... - copy references back to parent/coordination/management objects that might be unknown or write-only for the client code
Solution 3:
You can access private members of a class from within the class, even those of another instance.