1. Thinking and explanation of inheritance in OO
Inheritance contains the implication that all methods implemented in the parent class are in effect creating specifications and contracts. Although it does not force all subclasses to follow these contracts, if the subclass arbitrarily modifies the implemented methods, it will break the inheritance system.
*** inheritance brings convenience to programming as well as disadvantages. *** For example, using inheritance will bring invasion to the program, reduce the portability of the program and increase the coupling between objects. If a class is inherited by other classes, when the class needs to be modified, all subclasses must be considered, and after the parent class is modified, all functions involving subclasses may fail
Question posed: how to use inheritance correctly in programming? => Richter’s substitution principle
2. Basic introduction
-
The Liskov Substitution Principle was coined in 1988 by a lady named Lee at MIT.
-
If for every object o1 of type T1 there is an object O2 of type T2, such that the behavior of all programs P defined by T1 does not change when all objects o1 are substituted for O2, then type T2 is a subtype of type T1. In other words, all references to a base class must be able to transparently use objects from its subclasses.
-
When using inheritance, follow the Richter’s substitution rule and try not to override the parent method *** in *** subclasses
-
Richter’s substitution principle tells us that inheritance actually makes the two classes more coupled, and that aggregation, composition, and dependency can be used to solve problems *** where appropriate
3. A program leads to questions and reflections
/ / A class
class A {
// Return the difference between two numbers
public int func1(int num1, int num2) {
returnnum1 - num2; }}// Class B inherits class A
// Added a new function: add two numbers and then add them to 9
class B extends A {
// Here, the class A method is overridden, probably unconsciously
public int func1(int a, int b) {
return a + b;
}
public int func2(int a, int b) {
return func1(a, b) + 9; }}Copy the code
test
public static void main(String[] args) {
// TODO Auto-generated method stub
A a = new A();
System.out.println("11-3 =" + a.func1(11.3));
System.out.println("1-8 =" + a.func1(1.8));
System.out.println("-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --");
B b = new B();
System.out.println("11-3 =" + b.func1(11.3));// The original intention here is to solve for 11-3
System.out.println("1-8 =" + b.func1(1.8));/ / 1-8
System.out.println("11 + 3 + 9 =" + b.func2(11.3));
}
Copy the code
4. Solutions
-
We found an error in the subtraction function that worked well. The reason is that class B inadvertently overrides the methods of its parent class, causing the original functionality to fail. In practical programming, we often override the methods of the parent class to accomplish the new function, which is simple to write, but the overall inheritance system will be less reusable. Especially if you’re running a lot of polymorphism
-
The general approach is: the original parent class and child class inherit a more popular base class, the original inheritance relationship removed, replaced by dependency, aggregation, composition and other relations.
-
The improved scheme
5. Code implementation
// Create a more basic base class
class Base {
// Write more basic methods and members to the Base class
}
/ / A class
class A extends Base {
// Return the difference between two numbers
public int func1(int num1, int num2) {
returnnum1 - num2; }}// Class B inherits class A
// Added a new function: add two numbers and then add them to 9
class B extends Base {
// If B needs to use class A methods, use composition relationships
private A a = new A();
// Here, the class A method is overridden, probably unconsciously
public int func1(int a, int b) {
return a + b;
}
public int func2(int a, int b) {
return func1(a, b) + 9;
}
// We still want to use A's method
public int func3(int a, int b) {
return this.a.func1(a, b); }}Copy the code
test
public static void main(String[] args) {
// TODO Auto-generated method stub
A a = new A();
System.out.println("11-3 =" + a.func1(11.3));
System.out.println("1-8 =" + a.func1(1.8));
System.out.println("-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --");
B b = new B();
// Because class B no longer inherits from class A, func1 is no longer subtracted by the caller
// The finished function will be clear
System.out.println("11 + 3 =" + b.func1(11.3));// We want to find 11+3
System.out.println("1 + 8 =" + b.func1(1.8));/ / 1 + 8
System.out.println("11 + 3 + 9 =" + b.func2(11.3));
// Class A related methods can still be used with combinations
System.out.println("11-3 =" + b.func3(11.3));// The original intention here is to solve for 11-3
}
Copy the code