C ++ वंशानुक्रम

ऑब्जेक्ट-ओरिएंटेड प्रोग्रामिंग में सबसे महत्वपूर्ण अवधारणाओं में से एक विरासत की है। वंशानुक्रम हमें एक वर्ग को दूसरी कक्षा के संदर्भ में परिभाषित करने की अनुमति देता है, जिससे एप्लिकेशन बनाना और उसे बनाए रखना आसान हो जाता है। यह भी कोड कार्यक्षमता और तेजी से कार्यान्वयन समय का पुन: उपयोग करने का अवसर प्रदान करता है।

क्लास बनाते समय, पूरी तरह से नए डेटा सदस्यों और सदस्य कार्यों को लिखने के बजाय, प्रोग्रामर यह नामित कर सकता है कि नए वर्ग को किसी मौजूदा क्लास के सदस्यों को विरासत में प्राप्त करना चाहिए। इस मौजूदा वर्ग को कहा जाता हैbase वर्ग, और नए वर्ग के रूप में जाना जाता है derived कक्षा।

विरासत का विचार लागू करता है is aरिश्ते। उदाहरण के लिए, स्तनपायी IS-A पशु, कुत्ता IS-A स्तनपायी इसलिए कुत्ता IS-A पशु भी है।

आधार और व्युत्पन्न वर्ग

एक वर्ग को एक से अधिक वर्गों से प्राप्त किया जा सकता है, जिसका अर्थ है कि यह कई आधार वर्गों से डेटा और फ़ंक्शंस प्राप्त कर सकता है। एक व्युत्पन्न वर्ग को परिभाषित करने के लिए, हम बेस क्लास (तों) को निर्दिष्ट करने के लिए एक वर्ग व्युत्पत्ति सूची का उपयोग करते हैं। एक वर्ग व्युत्पत्ति सूची में एक या अधिक आधार वर्गों के नाम होते हैं और उनका फॉर्म होता है -

class derived-class: access-specifier base-class

जहां पहुंच-विनिर्देशक में से एक है public, protected, या private, और बेस-क्लास पहले से परिभाषित क्लास का नाम है। यदि पहुंच-विनिर्देश का उपयोग नहीं किया जाता है, तो यह डिफ़ॉल्ट रूप से निजी है।

बेस क्लास पर विचार करें Shape और इसका व्युत्पन्न वर्ग Rectangle निम्नानुसार है -

#include <iostream>
 
using namespace std;

// Base class
class Shape {
   public:
      void setWidth(int w) {
         width = w;
      }
      void setHeight(int h) {
         height = h;
      }
      
   protected:
      int width;
      int height;
};

// Derived class
class Rectangle: public Shape {
   public:
      int getArea() { 
         return (width * height); 
      }
};

int main(void) {
   Rectangle Rect;
 
   Rect.setWidth(5);
   Rect.setHeight(7);

   // Print the area of the object.
   cout << "Total area: " << Rect.getArea() << endl;

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Total area: 35

अभिगम नियंत्रण और विरासत

एक व्युत्पन्न वर्ग अपने आधार वर्ग के सभी गैर-निजी सदस्यों तक पहुँच सकता है। इस प्रकार बेस क्लास के सदस्य जो व्युत्पन्न वर्गों के सदस्य कार्यों के लिए सुलभ नहीं होने चाहिए, उन्हें बेस क्लास में निजी घोषित किया जाना चाहिए।

हम निम्नलिखित के अनुसार अलग-अलग पहुंच प्रकारों को संक्षेप में प्रस्तुत कर सकते हैं - जो उन्हें निम्नलिखित तरीके से एक्सेस कर सकते हैं -

पहुंच जनता संरक्षित निजी
एक ही कक्षा हाँ हाँ हाँ
व्युत्पन्न वर्ग हाँ हाँ नहीं
बाहर की कक्षाएं हाँ नहीं नहीं

एक व्युत्पन्न वर्ग को निम्नलिखित अपवादों के साथ सभी आधार वर्ग विधियाँ प्राप्त होती हैं -

  • बेस क्लास के निर्माता, विध्वंसक और कॉपी कंस्ट्रक्टर।
  • बेस क्लास के ओवरलोडेड ऑपरेटर।
  • आधार वर्ग का मित्र कार्य करता है।

वंशानुक्रम का प्रकार

जब एक वर्ग को आधार वर्ग से निकाला जाता है, तो आधार वर्ग को विरासत में प्राप्त किया जा सकता है public, protected या privateविरासत। विरासत का प्रकार ऊपर बताए अनुसार पहुंच-निर्दिष्ट द्वारा निर्दिष्ट किया गया है।

हम शायद ही उपयोग करते हैं protected या private विरासत, लेकिन publicवंशानुक्रम का आमतौर पर उपयोग किया जाता है। विभिन्न प्रकार की विरासत का उपयोग करते समय, निम्नलिखित नियम लागू होते हैं -

  • Public Inheritance - जब एक वर्ग व्युत्पन्न से एक public आधार वर्ग, public बेस क्लास के सदस्य बन जाते हैं public व्युत्पन्न वर्ग के सदस्य और protected बेस क्लास के सदस्य बन जाते हैं protectedव्युत्पन्न वर्ग के सदस्य। एक बेस क्लास काprivate सदस्य कभी भी एक व्युत्पन्न वर्ग से सीधे नहीं मिल सकते हैं, लेकिन कॉल के माध्यम से पहुँचा जा सकता है public तथा protected बेस क्लास के सदस्य।

  • Protected Inheritance - जब एक से व्युत्पन्न protected आधार वर्ग, public तथा protected बेस क्लास के सदस्य बन जाते हैं protected व्युत्पन्न वर्ग के सदस्य।

  • Private Inheritance - जब एक से व्युत्पन्न private आधार वर्ग, public तथा protected बेस क्लास के सदस्य बन जाते हैं private व्युत्पन्न वर्ग के सदस्य।

एकाधिक वंशानुक्रम

C ++ वर्ग एक से अधिक वर्ग के सदस्यों को विरासत में दे सकता है और यहाँ विस्तारित सिंटैक्स है -

class derived-class: access baseA, access baseB....

जहां पहुंच एक है public, protected, या privateऔर हर आधार वर्ग के लिए दिया जाएगा और वे ऊपर दिखाए गए अनुसार अल्पविराम से अलग हो जाएंगे। आइए हम निम्नलिखित उदाहरण की कोशिश करते हैं -

#include <iostream>
 
using namespace std;

// Base class Shape
class Shape {
   public:
      void setWidth(int w) {
         width = w;
      }
      void setHeight(int h) {
         height = h;
      }
      
   protected:
      int width;
      int height;
};

// Base class PaintCost
class PaintCost {
   public:
      int getCost(int area) {
         return area * 70;
      }
};

// Derived class
class Rectangle: public Shape, public PaintCost {
   public:
      int getArea() {
         return (width * height); 
      }
};

int main(void) {
   Rectangle Rect;
   int area;
 
   Rect.setWidth(5);
   Rect.setHeight(7);

   area = Rect.getArea();
   
   // Print the area of the object.
   cout << "Total area: " << Rect.getArea() << endl;

   // Print the total cost of painting
   cout << "Total paint cost: $" << Rect.getCost(area) << endl;

   return 0;
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Total area: 35
Total paint cost: $2450