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.