Do Sub-Classes Really Inherit Private Member Variables?
Asked Answered
A

7

13

Basically as far as I know, when you create a base class with a public, protected, and private section and variables/functions in each the public and protected sections will get inherited into the appropriate section of the sub-class (defined by class subclass : private base, which will take all public and protected members of base and put them into public, changing the word private to public puts them all in public and changing it to protected puts them all into protected).

So, when you create a sub-class you never receive anything from the private section of the previous class (the base class in this case), if this is true then an object of the sub-class should never have it's own version of a private variable or function from the base class correct?

Let's run over an example:

#include <iostream>

class myClass     // Creates a class titled myClass with a public section and a private section.
{
public:
  void setMyVariable();
  int getMyVariable();
private:
  int myVariable;     // This private member variable should never be inherited.
};

class yourClass : public myClass {};    // Creates a sub-class of myClass that inherits all the public/protected members into  the
// public section of yourClass. This should only inherit setMyVariable()
// and getMyVariable() since myVariable is private. This class does not over-ride any
// functions so it should be using the myClass version upon each call using a yourClass
// object. Correct?

int main()
{
  myClass myObject;           // Creates a myClass object called myObject.
  yourClass yourObject;       // Creates a yourClass object called yourObject
  yourObject.setMyVariable(); // Calls setMyVariable() through yourObject. This in turn calls the myClass version of it    because
  // there is no function definition for a yourClass version of this function. This means that this
  // can indeed access myVariable, but only the myClass version of it (there isn't a yourClass
  // version because myVariable is never inherited).

  std::cout << yourObject.getMyVariable() << std::endl;   // Uses the yourClass version of getMyVariable() which in turn
  // calls the myClass version, thus it returns the myClass myVariable
  // value. yourClass never has a version of myVariable Correct?

  std::cout << myObject.getMyVariable() << std::endl;     // Calls the myClass version of getMyVariable() and prints myVariable.

  return 0;
}

void myClass::setMyVariable()
{
  myVariable = 15;        // Sets myVariable in myClass to 15.
}

int myClass::getMyVariable()
{
  return myVariable;      // Returns myVariable from myClass.
}

Now, in theory based on what I think, this should print: 15 15 Due to it simply always using the myClass version of the functions (thus using the myClass myVariable). But, strangely, this is not the case. The result of running this program prints: 15 0 This makes me wonder, are we actually not only inheriting myVariable, but do we also have the ability to mess around with it? Clearly this is creating an alternate version of myVariable somehow otherwise there wouldn't be a 0 for the myClass version. We are indeed editing a second copy of myVariable by doing all this.

Can someone please explain this all to me, this has torn apart my understanding of inheritance.

Ahern answered 11/1, 2013 at 2:16 Comment(4)
Could you please fix the indentation of your code?Configuration
myObject.getMyVariable() is undefined behaviour.Martimartial
You never call myObject.setMyVariable. So it's a simple coding bug; nothing to do with inheritance!Configuration
I think it may be some confusion about the distinction between classes and objects and the misconception that this distinction somehow vanishes as soon as you use inheritance.Clarita
E
20

Basically as far as I know, when you create a base class with a public, protected, and private section and variables/functions in each the public and protected sections will get inherited into the appropriate section of the sub-class (defined by class subclass : private base, which will take all public and private members of base and put them into public, changing the word private to public puts them all in public and changing it to protected puts them all into protected).

There's a bit of confusion in this statement.

Recall that inheritance is defined for classes and structs in C++. Individual objects (ie. instances) do not inherit from other objects. Constructing an object using other objects is called composition.

When a class inherits from another class, it gets everything from that class, but the access level of the inherited fields may inhibit their use within the inheritor.

Furthermore, there are 3 kinds of inheritance for classes: private (which is the default), protected, and public. Each of them changes the access level of a class properties and methods when inherited by a subclass.

If we order the access levels in this manner: public, protected, private, from the least protected to the most protected, then we can define the inheritance modifiers as raising the access levels of the inherited class fields to at least the level they designate, in the derived class (ie. the class inheriting).

For instance, if class B inherits from class A with the protected inheritance modifier:

  class B : protected A { /* ... */ };

then all the fields from A will have at least the protected level in B:

  • public fields become protected (public level is raised to protected),
  • protected fields stay protected (same access level, so no modification here),
  • private fields stay private (the access level is already above the modifier)
Economizer answered 11/1, 2013 at 3:3 Comment(1)
Wow, thank you very much for the useful post. That explained everything perfectly! (thank you all really but this one got the best answer)Ahern
A
11

"When you create a sub-class you never receive anything from the private section of the [base class]. If this is true then an object of the sub-class should never have it's own version of a private variable or function from the base class, correct?"

No. The derived class inherits all the members of the base class, including the private ones. An object of the inherited class has those private members, but does not have direct access to them. It has access to public members of the base class that may have access to those members, but it (the derived class) may not have new member functions with such access:

class yourClass : public myClass
{
public:
  void playByTheRules()
  {
    setMyVariable(); // perfectly legal, since setMyVariable() is public
  }

  void tamperWithMyVariable()
  {
    myVariable = 20; // this is illegal and will cause a compile-time error
  }
};
Adverbial answered 11/1, 2013 at 2:35 Comment(5)
Interesting, so can you access the private members that were inherited through an alternative method? I mean, can I access the myVariable in yourClass in anyway? I see what you mean, but wouldn't getMyVariable just access the base class's version of myVariable instead of accessing the yourclass version? I've read that you cannot use any private member variables that are inherited into the sub-class (researched more after your post).Ahern
You can't use them in the implementation of the subclass. You can of course use them in the implementation of the base class though, and call public methods/access public fields of the base class from the outside.Clarita
Right right, but what I mean is, how can you access and modify the sub-class's version of the private members it inherits?Ahern
There is only one version of the private member. The subclass inherits the member from the base class, there is nothing like a second copy. Behind the scenes, what you're actually accessing and modifying is the base class's version. (This does not mean that your yourObject is somehow accessing myObject, they are different objects!)Clarita
Ahh so it is similar to inheriting a public function, you gain access to it (though in this case not directly) but you don't get your own version of it unless your over-ride it?Ahern
C
7

myObject and yourObject are two different objects! Why should they share anything?

Think about it that way: Forget about inheritance and suppose you have a class Person with private int age; and public void setAge (int age) {...}. You then instantiate two objects:

Person bob;
Person bill;
bob.setAge(35);

Would you expect Bill to be 35 now, too? You wouldn't, right? Similarly, your myObject doesn't share its data with yourObject.


In response to your comment:

The class yourClass inherits from myClass. That means that both yourObject and myObject have their own myVariable, the latter obviously by definition, the former inherited from myClass.

Clarita answered 11/1, 2013 at 2:23 Comment(3)
OH!!!!! No wonder, it is two different object, that's why! Each object has it's own instance of the functions and such. So am I right that myVariable is never inherited? I just want to clear that up, is my understanding of inheritance correct?Ahern
It is inherited, see Beta's answer!Clarita
Thank you very much for the little addition there, everything all makes sense now. Thank you too!Ahern
F
6

Physically, every single member( including member functions) of base class goes into the subclass. Doesn't matter if they are private. Doesn't matter if you inherit them publically/protected-ly/privately. So in your example, yourClass contains all three of getMyVariable(), setMyVariable() and myVariable. All this is pretty simple, okay?

What matters is how we can access them. It is like when a file is deleted on your system. So, you should first understand the difference between a member being not there and a member being there but inaccessible. Assume for now that all inheritance takes place publically. Then, all public members of base class are public in derived class, protected members are protected and private members are inaccessible. They are inaccessible and not non-existent because there can be some member functions in protected and public sections in base class which access the private members of base class. Thus, we need all those private members of base which are accessed by public and protected member functions of base, for their functionality. Since there is no way that we can determine which member is needed by which member function in a simple manner, we include all private members of the base class in derived class. All this simply means that in a derived class, a private member can be modified by only through the base class' member functions.

Note: every private member has to be accessed, directly or indirectly [through another private member function which in turn is called by a public/protected member function] by a public/protected meber function, else it has no use.

So, we know till now that a private member variable of base class has its use in derived class i.e. for the functionality of its public/protected member functions. But they can't be accessed directly in base class.

Now, we turn our attention to private/public inheritance. For public inheritance, it means that all the accessible members of base class (that is, the public and protected members) can not be at a level more permissive than public. Since, public is the most permissive level, public and protected members remain public. But at protected and private inheritance, both become protected and private in the derived class, respectively. Inthe latter case, since all these members are private, they can't be accessed further in the hierarchy chain, but can be accessed by the given derived class all the same.

Thus, the level of each base class member in derived class is the lesser of their level in derived class () and the type of inheritance (public/protected/private).

Same concept applies to the functions outside the class. For them private and protected members are inaccessible but they do exist and can be accessed by the public member functions.

And taking your case as a final example, setMyvariable() and getMyVariable() can access myVariable in the derived class. But no function specified in derived class can access myVariable. Modifying your class:

class myClass
{
public:
  void setMyVariable();
  int getMyVariable();
private:
  int myVariable;
};

class yourClass : public myClass
{
public:
  // void yourFunction() { myVariable = 1; }
  /*Removing comment creates error; derived class functions can't access myVariable*/
};

Further: you can add exceptions to the type of inheritance too e.g. a private inheritance except a member made public in derived class. But that is another question altogether.

Fir answered 11/1, 2013 at 3:17 Comment(1)
Great answer ! Very clear & covered every aspect of question.Obduliaobdurate
C
5

You never call myObject.setMyVariable(), so myObject.getMyVariable() will not return 15.

private does not imply static.

Configuration answered 11/1, 2013 at 2:21 Comment(2)
Well, it cooould... I know what you mean, though :)Martimartial
That's the point of this question, why is it that yourObject.getMyVariable returns 15? This means there are two instances of myVariable, otherwise no matter which object you used to set/get it would return 15 on both. Clearly there are two myVariables somehow, that is the question.Ahern
E
0

After:

class yourClass : public myClass {};

there is still only one member variable. But there are two ways of accessing it by name: myClass::myVariable, and yourClass::myVariable.

In these expressions, the class name is known as the naming class. The second key thing to understand is that access rights apply to the combination of naming class and member name; not just to the member name and not to the variable itself.

If a member is mentioned without explicitly having the naming class present, then the naming class is inferred from the type of the expression on the left of the . or -> that named the member (with this-> being implied if there is no such expression).

Furthermore, there are really four possible types of access: public, protected, private, and no access. You can't declare a member as having no access, but that situation arises when a private member is inherited.


Applying all this theory to your example:

  • The name myClass::myVariable is private.
  • The name yourClass::myVariable is no access.

To reiterate, there is only actually one variable, but it may be named in two different ways, and the access rights differ depending on which name is used.


Finally, back to your original example. myObject and yourObject are different objects. I think what you intended to write, or what you are mentally imagining is actually this situation:

yourClass yourObject;
myClass& myObject = yourObject;
//    ^^^

which means myObject names the base class part of yourObject. Then after:

yourObject.setMyVariable();

the variable is set to 15, and so

std::cout << myObject.getMyVariable() << std::endl;

would output 15 because there is indeed only one variable.

Ellary answered 1/9, 2016 at 8:12 Comment(0)
M
-2

This may help

#include<iostream>
using namespace std;

class A
{
int b;  
};

class B : private A
{

};

int main()
{
C obj;
cout<<sizeof(obj);  
return 0;
}
Mylohyoid answered 12/9, 2017 at 4:29 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.