Java - ความหลากหลาย
Polymorphism คือความสามารถของวัตถุในหลายรูปแบบ การใช้ความหลากหลายของความหลากหลายใน OOP เกิดขึ้นเมื่อการอ้างอิงคลาสพาเรนต์ถูกใช้เพื่ออ้างถึงอ็อบเจ็กต์คลาสลูก
ออบเจ็กต์ Java ใด ๆ ที่สามารถผ่านการทดสอบ IS-A ได้มากกว่าหนึ่งรายการจะถือว่าเป็นโพลีมอร์ฟิก ใน Java อ็อบเจ็กต์ Java ทั้งหมดเป็นโพลีมอร์ฟิกเนื่องจากอ็อบเจ็กต์ใด ๆ จะผ่านการทดสอบ IS-A สำหรับประเภทของตนเองและคลาสอ็อบเจ็กต์
สิ่งสำคัญคือต้องรู้ว่าวิธีเดียวที่เป็นไปได้ในการเข้าถึงวัตถุคือผ่านตัวแปรอ้างอิง ตัวแปรอ้างอิงสามารถมีได้เพียงประเภทเดียว เมื่อประกาศแล้วจะไม่สามารถเปลี่ยนประเภทของตัวแปรอ้างอิงได้
ตัวแปรอ้างอิงสามารถกำหนดใหม่ให้กับออบเจ็กต์อื่น ๆ ได้หากไม่ได้ประกาศขั้นสุดท้าย ชนิดของตัวแปรอ้างอิงจะกำหนดวิธีการที่สามารถเรียกใช้กับวัตถุ
ตัวแปรอ้างอิงสามารถอ้างถึงออบเจ็กต์ใด ๆ ในประเภทที่ประกาศหรือประเภทย่อยใด ๆ ของประเภทที่ประกาศ ตัวแปรอ้างอิงสามารถประกาศเป็นคลาสหรือประเภทอินเทอร์เฟซได้
ตัวอย่าง
ให้เราดูตัวอย่าง
public interface Vegetarian{}
public class Animal{}
public class Deer extends Animal implements Vegetarian{}
ตอนนี้คลาส Deer ถือเป็นความหลากหลายเนื่องจากมีการสืบทอดหลายแบบ ต่อไปนี้เป็นจริงสำหรับตัวอย่างข้างต้น -
- กวางเป็นสัตว์
- กวางเป็นมังสวิรัติ
- กวางคือกวาง
- กวางเป็นวัตถุ
เมื่อเราใช้ข้อเท็จจริงของตัวแปรอ้างอิงกับการอ้างอิงอ็อบเจ็กต์ Deer การประกาศต่อไปนี้ถือเป็นกฎหมาย -
ตัวอย่าง
Deer d = new Deer();
Animal a = d;
Vegetarian v = d;
Object o = d;
ตัวแปรอ้างอิงทั้งหมด d, a, v, o อ้างถึงอ็อบเจกต์ Deer เดียวกันในฮีป
วิธีการเสมือนจริง
ในส่วนนี้ฉันจะแสดงให้คุณเห็นว่าพฤติกรรมของวิธีการแทนที่ใน Java ช่วยให้คุณใช้ประโยชน์จากความหลากหลายในการออกแบบคลาสของคุณได้อย่างไร
เราได้พูดถึงการลบล้างเมธอดแล้วซึ่งคลาสลูกสามารถแทนที่เมธอดในพาเรนต์ได้ เมธอดที่ถูกแทนที่ถูกซ่อนอยู่ในคลาสพาเรนต์เป็นหลักและจะไม่ถูกเรียกใช้เว้นแต่คลาสลูกจะใช้ซูเปอร์คีย์เวิร์ดภายในเมธอดการลบล้าง
ตัวอย่าง
/* 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 ()คอมไพลเลอร์จะเห็น mailCheck () ในคลาส Salary ในเวลาคอมไพล์และ JVM จะเรียกใช้ mailCheck () ในคลาส Salary ในขณะรัน
mailCheck () บน e ค่อนข้างแตกต่างกันเพราะ eเป็นข้อมูลอ้างอิงของพนักงาน เมื่อคอมไพลเลอร์เห็นe.mailCheck ()คอมไพเลอร์จะเห็นเมธอด mailCheck () ในคลาส Employee
ในเวลาคอมไพล์คอมไพเลอร์ใช้ mailCheck () ใน Employee เพื่อตรวจสอบคำสั่งนี้ อย่างไรก็ตามในขณะดำเนินการ JVM จะเรียกใช้ mailCheck () ในคลาส Salary
ลักษณะการทำงานนี้เรียกว่าการเรียกใช้เมธอดเสมือนและวิธีการเหล่านี้เรียกว่าเมธอดเสมือน มีการเรียกใช้เมธอดที่ถูกแทนที่ในขณะรันไม่ว่าจะใช้ข้อมูลอ้างอิงประเภทใดในซอร์สโค้ดในเวลาคอมไพล์