Iāll try to give you primer on this, although it is more suited for a training course and I do not make a good instructor 
Inheritance
The idea of inheritance is that particular sets of ābehaviorā and āpropertiesā are not limited to one object but are more general. For and example, letās look at pets. For the sake of clarity, Iāll assume that the only pets are cats and dogs (otherwise it will get too complicated). General behaviors of these pets are eating, moving, making sounds, showing affection. General properties are fur-type, tail-length, ear-shape, healthiness. You could create to separate objects ācatā and ādogā, but then you would repeat the same stuff a lot. Also it would be nice you could take some pet (dog or cat) to the vet instead of explicitly take a dog to the vet or a cat to the vet. Thatās where inheritance can be nice:
Iāll try to do some pseudo object description language:
pet:
// Behaviors
doEat
doMove
doMakeSound
doShowAffection
// Properties
furType // Curly, Straight, Bald
tailLength // In inches
earShape // Pointy, Rounded
isHealthy // Yes/No
cat (is a pet): // Inherit from pet
doEat
digest cat food
doMove
walk without the nails making a sound
doMakeSound
perform meow
doShowAffection
give headbutts
dog (is a pet): // Inherit from pet
doEat
digest dog food
doMove
walk with ticking nails
doMakeSound
bark
doShowAffection
lick
In the above example, you can see the set of behaviors and properties are the same for both cat and dog, but the actual way the behaviors work are differently āimplementedā
The power of this is shown by this:
pet myPets[3]
pet.add(new dog)
pet.add(new cat)
pet.add(new cat)
for(pet somePet in myPets)
somePet.doMakeSound
In this example while iterating through the array of myPets
each pet will make its distinctive sound, due to their implementation.
In c++ it would look like this:
class pet
{
public:
virtual void doMakeSound() = 0;
virtual void doShowAffection() = 0;
int tailLenght = 0;
bool isHealty = true;
// ...
};
class cat: public pet
{
cat(int tailLength /*, other properties */)
void doMakeSound() override
{
// Play meow-wave
}
void doShowAffection() override
{
// Show picture of cat doing headbutts
}
// ...
};
class dog: public pet
{
dog(int tailLength /*, other properties */)
void doMakeSound() override
{
// Play bark-wave
}
void doShowAffection() override
{
// Show picture of dog licking face of boss
}
// ...
};
void main()
{
pet myPets[3] = {dog(10), cat(0), cat(7)};
for(int x = 0; x < 3; ++1)
myPets[x].doMakeSound();
}
( I didnāt compile it, so it could be not completely syntactically right)
Terminology
In the above examples doShowAffection
, doMakeSound
and so on are called āmethodsā. Play bark-wave
, Play meow-wave
, Show picture [...]
and so on are the implementations of the methods āinheritedā from pet. The lines class cat : public pet
and class dog : public pet
express that inheritance.
In a class, if you need to refer to the instance a method is currently working for, you can use the word this
. So letās apply this to the pet example:
class vet
{
public:
void curePet(pet thePet, boss itsBoss)
{
// Do something that hopefully cures the pet
sendBillToBoss(itsBoss);
}
// ...
};
class boss
{
public:
vet myVet;
void checkPet(pet myPet)
{
if(!myPet.isHealty)
myVet.curePet(myPet, this);
}
};
Lambda
In programming languages you have various types of variables like int
, long
, string
and so on. Wouldnāt it be nice if there would be a type of variable that could be assigned a function of function body so that later on you could call that variable to execute that function? Pseudo language:
function makeSound;
void main()
{
makeSound = { setVolume(100); playWaveFile(1); }
makeSound();
makeSound = { setVolume(50); playWaveFile(7); }
makeSound();
makeSound = { switchOnTelevision(true); }
makeSound();
}
A lambda takes this idea to a higher level: You assign a function-body, but you can have it operating in the context of what is accessible (with regards to variables) at time you assign the lambda. Here is a snippet showing that:
void lambdaDemo(String text)
{
myText = text;
String someTitle = "Show data from MainComponent";
juce::Timer::callAfterDelay(3000,
// Lambda starts here
[this, someTitle] // List of variables that the body has access to
{ // Body of the lambda
juce::AlertWindow::showMessageBox(AlertWindow::AlertIconType::InfoIcon, someTitle, myText);
} // End of the body
); // This belongs to `juce::Timer:callAfterDelay(...)`
someTitle = "This title will not be show";
}
myText
is a string variable defined in the class MainComponent
. Because this
is in the list of variables, the body can access it. The variable someTitle
is a local variable, but also seems available to the lambda body, even while the function has already exited, but actually it gets a copy.
Conclusion
I hope this did explain some things and I didnāt confuse things any further (āI would have understood it, if he wouldnāt have explained itā
);
Here is a simple juce 7.0.12 project showing the juce::Timer inheritance and the lambda:
GUIOnly.zip (17.0 KB)
Disclaimer
There may be errors in terminology, syntax and explanations. Also the demo project may contain errors (although it compiles and executes fine on my system)