Interfejsy w C ++ (klasy abstrakcyjne)
Interfejs opisuje zachowanie lub możliwości klasy C ++ bez angażowania się w konkretną implementację tej klasy.
Interfejsy C ++ są implementowane przy użyciu abstract classes i tych klas abstrakcyjnych nie należy mylić z abstrakcją danych, która jest koncepcją oddzielenia szczegółów implementacji od powiązanych danych.
Klasa jest abstrakcyjna, deklarując co najmniej jedną z jej funkcji jako pure virtualfunkcjonować. Czysta funkcja wirtualna jest określana przez umieszczenie w jej deklaracji „= 0” w następujący sposób -
class Box {
public:
// pure virtual function
virtual double getVolume() = 0;
private:
double length; // Length of a box
double breadth; // Breadth of a box
double height; // Height of a box
};
Celem abstract class(często określane jako ABC) ma na celu zapewnienie odpowiedniej klasy bazowej, z której inne klasy mogą dziedziczyć. Klasy abstrakcyjne nie mogą służyć do tworzenia instancji obiektów i służą tylko jako plikinterface. Próba utworzenia wystąpienia obiektu klasy abstrakcyjnej powoduje błąd kompilacji.
Tak więc, jeśli trzeba utworzyć instancję podklasy ABC, musi zaimplementować każdą z funkcji wirtualnych, co oznacza, że obsługuje interfejs zadeklarowany przez ABC. Niepowodzenie zastąpienia czystej funkcji wirtualnej w klasie pochodnej, a następnie próba utworzenia instancji obiektów tej klasy jest błędem kompilacji.
Klasy, których można użyć do tworzenia instancji obiektów, są wywoływane concrete classes.
Przykład klasy abstrakcyjnej
Rozważmy następujący przykład, w którym klasa nadrzędna udostępnia interfejs do klasy bazowej w celu zaimplementowania funkcji o nazwie getArea() -
#include <iostream>
using namespace std;
// Base class
class Shape {
public:
// pure virtual function providing interface framework.
virtual int getArea() = 0;
void setWidth(int w) {
width = w;
}
void setHeight(int h) {
height = h;
}
protected:
int width;
int height;
};
// Derived classes
class Rectangle: public Shape {
public:
int getArea() {
return (width * height);
}
};
class Triangle: public Shape {
public:
int getArea() {
return (width * height)/2;
}
};
int main(void) {
Rectangle Rect;
Triangle Tri;
Rect.setWidth(5);
Rect.setHeight(7);
// Print the area of the object.
cout << "Total Rectangle area: " << Rect.getArea() << endl;
Tri.setWidth(5);
Tri.setHeight(7);
// Print the area of the object.
cout << "Total Triangle area: " << Tri.getArea() << endl;
return 0;
}
Kiedy powyższy kod jest kompilowany i wykonywany, daje następujący wynik -
Total Rectangle area: 35
Total Triangle area: 17
Możesz zobaczyć, jak klasa abstrakcyjna zdefiniowała interfejs za pomocą metody getArea (), a dwie inne klasy zaimplementowały tę samą funkcję, ale z innym algorytmem obliczania obszaru specyficznego dla kształtu.
Projektowanie strategii
System zorientowany obiektowo może używać abstrakcyjnej klasy bazowej, aby zapewnić wspólny i znormalizowany interfejs odpowiedni dla wszystkich aplikacji zewnętrznych. Następnie, poprzez dziedziczenie z tej abstrakcyjnej klasy bazowej, tworzone są klasy pochodne, które działają podobnie.
Możliwości (tj. Funkcje publiczne) oferowane przez aplikacje zewnętrzne są dostarczane jako czyste funkcje wirtualne w abstrakcyjnej klasie bazowej. Implementacje tych czystych funkcji wirtualnych są dostarczane w klasach pochodnych, które odpowiadają określonym typom aplikacji.
Architektura ta umożliwia również łatwe dodawanie nowych aplikacji do systemu, nawet po zdefiniowaniu systemu.