जावा - बहुरूपता

बहुरूपता एक वस्तु की कई रूपों को लेने की क्षमता है। OOP में बहुरूपता का सबसे आम उपयोग तब होता है जब एक माता-पिता वर्ग के संदर्भ का उपयोग बाल वर्ग वस्तु के संदर्भ में किया जाता है।

कोई भी जावा ऑब्जेक्ट जो एक से अधिक IS-A परीक्षण पास कर सकता है, उसे बहुरूपी माना जाता है। जावा में, सभी जावा ऑब्जेक्ट पॉलीमॉर्फिक हैं क्योंकि कोई भी वस्तु अपने स्वयं के प्रकार और वर्ग वस्तु के लिए आईएस-ए परीक्षण पास करेगी।

यह जानना महत्वपूर्ण है कि किसी वस्तु तक पहुंचने का एकमात्र संभव तरीका एक संदर्भ चर है। एक संदर्भ चर केवल एक प्रकार का हो सकता है। एक बार घोषित करने के बाद, संदर्भ चर का प्रकार बदला नहीं जा सकता।

संदर्भ चर को अन्य वस्तुओं को फिर से असाइन किया जा सकता है बशर्ते कि इसे अंतिम घोषित नहीं किया गया हो। संदर्भ चर का प्रकार उन तरीकों को निर्धारित करेगा जो इसे ऑब्जेक्ट पर लागू कर सकते हैं।

एक संदर्भ चर अपने घोषित प्रकार के किसी भी ऑब्जेक्ट या इसके घोषित प्रकार के किसी भी उपप्रकार का उल्लेख कर सकता है। एक संदर्भ चर को एक वर्ग या इंटरफ़ेस प्रकार के रूप में घोषित किया जा सकता है।

उदाहरण

आइए एक उदाहरण देखें।

public interface Vegetarian{}
public class Animal{}
public class Deer extends Animal implements Vegetarian{}

अब, हिरण वर्ग को बहुपत्नी माना जाता है क्योंकि इसमें कई वंशानुक्रम होते हैं। उपरोक्त उदाहरणों के लिए निम्नलिखित सत्य हैं -

  • एक हिरण IS-A पशु है
  • एक हिरण IS-A शाकाहारी
  • एक हिरण IS-A हिरण
  • एक हिरण IS-A ऑब्जेक्ट

जब हम संदर्भ चर तथ्यों को एक हिरण ऑब्जेक्ट संदर्भ पर लागू करते हैं, तो निम्नलिखित घोषणाएं कानूनी हैं -

उदाहरण

Deer d = new Deer();
Animal a = d;
Vegetarian v = d;
Object o = d;

सभी संदर्भ चर d, a, v, o ढेर में समान हिरण ऑब्जेक्ट को संदर्भित करते हैं।

आभासी तरीके

इस खंड में, मैं आपको दिखाऊंगा कि कैसे जावा में ओवरराइड तरीकों का व्यवहार आपको इन वर्गों को डिजाइन करते समय बहुरूपता का लाभ उठाने की अनुमति देता है।

हमने पहले ही ओवरराइडिंग विधि पर चर्चा की है, जहां एक बच्चा वर्ग अपने माता-पिता में एक विधि को ओवरराइड कर सकता है। मूल कक्षा में एक ओवरराइड विधि अनिवार्य रूप से छिपी हुई है, और जब तक कि बच्चा वर्ग ओवरराइडिंग विधि के भीतर सुपर कीवर्ड का उपयोग नहीं करता है, तब तक इसे लागू नहीं किया जाता है।

उदाहरण

/* File name : Employee.java */
public class Employee {
   private String name;
   private String address;
   private int number;

   public Employee(String name, String address, int number) {
      System.out.println("Constructing an Employee");
      this.name = name;
      this.address = address;
      this.number = number;
   }

   public void mailCheck() {
      System.out.println("Mailing a check to " + this.name + " " + this.address);
   }

   public String toString() {
      return name + " " + address + " " + number;
   }

   public String getName() {
      return name;
   }

   public String getAddress() {
      return address;
   }

   public void setAddress(String newAddress) {
      address = newAddress;
   }

   public int getNumber() {
      return number;
   }
}

अब मान लीजिए कि हम कर्मचारी वर्ग का विस्तार करते हैं -

/* File name : Salary.java */
public class Salary extends Employee {
   private double salary; // Annual salary
   
   public Salary(String name, String address, int number, double salary) {
      super(name, address, number);
      setSalary(salary);
   }
   
   public void mailCheck() {
      System.out.println("Within mailCheck of Salary class ");
      System.out.println("Mailing check to " + getName()
      + " with salary " + salary);
   }
   
   public double getSalary() {
      return salary;
   }
   
   public void setSalary(double newSalary) {
      if(newSalary >= 0.0) {
         salary = newSalary;
      }
   }
   
   public double computePay() {
      System.out.println("Computing salary pay for " + getName());
      return salary/52;
   }
}

अब, आप निम्नलिखित कार्यक्रम का ध्यानपूर्वक अध्ययन करें और इसके आउटपुट को निर्धारित करने का प्रयास करें -

/* File name : VirtualDemo.java */
public class VirtualDemo {

   public static void main(String [] args) {
      Salary s = new Salary("Mohd Mohtashim", "Ambehta, UP", 3, 3600.00);
      Employee e = new Salary("John Adams", "Boston, MA", 2, 2400.00);
      System.out.println("Call mailCheck using Salary reference --");   
      s.mailCheck();
      System.out.println("\n Call mailCheck using Employee reference--");
      e.mailCheck();
   }
}

यह निम्नलिखित परिणाम का उत्पादन करेगा -

उत्पादन

Constructing an Employee
Constructing an Employee

Call mailCheck using Salary reference --
Within mailCheck of Salary class
Mailing check to Mohd Mohtashim with salary 3600.0

Call mailCheck using Employee reference--
Within mailCheck of Salary class
Mailing check to John Adams with salary 2400.0

यहाँ, हम दो वेतन वस्तुओं को पलटें। एक वेतन संदर्भ का उपयोग करs, और एक कर्मचारी संदर्भ का उपयोग कर अन्य e

S.mailCheck () को लागू करते समय , कंपाइलर मेल का चेक () संकलन समय पर वेतन वर्ग में देखता है, और JVM रन टाइम में मेलचेक () को वेतन वर्ग में आमंत्रित करता है।

mailCheck () पर e काफी अलग है क्योंकि eएक कर्मचारी संदर्भ है। जब कंपाइलर e.mailCheck () देखता है , तो कंपाइलर कर्मचारी वर्ग में mailCheck () विधि देखता है।

यहाँ, संकलन के समय, संकलक ने इस कथन को मान्य करने के लिए कर्मचारी में mailCheck () का उपयोग किया। हालांकि, रन टाइम में, JVM सैलरी क्लास में mailCheck () को आमंत्रित करता है।

इस व्यवहार को वर्चुअल मेथड इनवोकेशन के रूप में संदर्भित किया जाता है, और इन विधियों को वर्चुअल मेथड के रूप में संदर्भित किया जाता है। रन समय में एक ओवरराइड विधि को लागू किया जाता है, कोई भी बात नहीं है कि डेटा प्रकार का संदर्भ क्या है जो संकलन समय पर स्रोत कोड में उपयोग किया गया था।