Nạp chồng phương thức (method overloading)
Nạp chồng phương thức trong java xảy ra nếu một lớp có nhiều phương thức có tên giống nhau nhưng khác nhau về kiểu dữ liệu hoặc số lượng các tham số.
Giả sử bạn phải thực hiện tính tổng của các số đã cho với bất kỳ số lượng các đối số, nếu bạn viết phương thức a(int, int) cho 2 tham số, b(int, int, int) cho 3 tham số điều này có thể gây khó hiểu cho các lập trình viên khác về ý nghĩa của các phương thức đó vì chúng có tên khác nhau.
Lợi ích của nạp chồng phương thức
Sử dụng nạp chồng phương thức giúp tăng khả năng đọc hiểu chương trình.
Các cách nạp chồng phương thức
Có 2 cách nạp chồng phương thức trong java
- Thay đổi số lượng các tham số
- Thay đổi kiểu dữ liệu của các tham số
1. Nạp chồng phương thức: thay đổi số lượng các tham số
Trong ví dụ này, chúng ta cần tạo 2 phương thức, phương thức add() đầu tiên thực hiện việc tính tổng của 2 số, phương thức thứ hai thực hiện việc tính tổng của 3 số. Sử dụng phương thức static để gọi hàm thông qua tên class thay vì phải tạo thể hiên của lớp.
class Adder{ static int add(int a,int b){return a+b;} static int add(int a,int b,int c){return a+b+c;} } class TestOverloading1{ public static void main(String[] args){ System.out.println(Adder.add(11,11)); System.out.println(Adder.add(11,11,11)); } }
Kết quả:
22 33
2. Nạp chồng phương thức: thay đổi kiểu dữ liệu của các tham số
Trong ví dụ này, chúng ta sẽ tạo ra 2 phương thức có kiểu dữ liệu khác nhau. Phương thức add() đầu tiên nhận 2 đổi số có kiểu giá trị là integer, phương thức thứ hai nhận 2 đổi số có kiểu giá trị là double.
class Adder{ static int add(int a, int b){return a+b;} static double add(double a, double b){return a+b;} } class TestOverloading2{ public static void main(String[] args){ System.out.println(Adder.add(11,11)); System.out.println(Adder.add(12.3,12.6)); } }
Kết quả:
22 24.9
Các câu hỏi về nạp chồng phương thức trong java
Câu hỏi 1: Tại sao không thể nạp chồng phương thức bằng cách chỉ thay đổi kiểu trả về của phương thức?
Trong java, không thể nạp chồng phương thức bằng cách chỉ thay đổi kiểu trả về của phương thức bởi vì không biết phương thức nào sẽ được gọi.
Ví dụ:
class Adder{ static int add(int a,int b){return a+b;} static double add(int a,int b){return a+b;} } class TestOverloading3{ public static void main(String[] args){ System.out.println(Adder.add(11,11));//ambiguity } }
Kết quả:
Compile Time Error: method add(int,int) is already defined in class Adder
Câu hỏi 2: Có thể nạp chồng phương thức main() không?
Có, bạn có thể nạp chồng n phương thức main. Nhưng JVM chỉ gọi phương thức main() có tham số truyền vào là một mảng String. Ví dụ:
public class TestOverloading4 { public static void main(String[] args) { System.out.println("main with String[]"); } public static void main(String args) { System.out.println("main with String"); } public static void main() { System.out.println("main without args"); } }
Kết quả:
main with String[]
Nạp chồng phương thức và sự thay đổi kiểu giá trị
Kiểu dữ liệu của đối số truyền vào được thay đổi sang kiểu dữ liệu khác (tự động ép kiểu) nếu giá trị của đối số đó không phù hợp với kiểu dữ liệu của tham số đã được đinh nghĩa. Để hiểu khái niệm này hãy xem ảnh sau:
Kiểu byte có thể được ép sang các kiểu short, int, long, float hoặc double. Kiểu dữ liệu short có thể được ép sang các kiểu int, long, float hoặc double. Kiểu dữ liệu char có thể được ép sang các kiểu int, long, float or double…
Ví dụ 1:
public class OverloadingCalculation1 { void sum(int a, long b) { System.out.println(a + b); } void sum(int a, int b, int c) { System.out.println(a + b + c); } public static void main(String args[]) { OverloadingCalculation1 obj = new OverloadingCalculation1(); // kiểu integer tham số 2 sẽ được thay đổi thành kiểu long obj.sum(20, 20); obj.sum(20, 20, 20); } }
Kết quả:
40 60
Ví dụ 2: nếu không có kiểu đối số nào phù hợp, chuyển đổi kiểu sẽ không được thực hiện.
public class OverloadingCalculation2 { void sum(int a, int b) { System.out.println("int arg method invoked"); } void sum(long a, long b) { System.out.println("long arg method invoked"); } public static void main(String args[]) { OverloadingCalculation2 obj = new OverloadingCalculation2(); // phương thứ sum() có đối số int sẽ được gọi obj.sum(20, 20); } }
Kết quả:
int arg method invoked
Ví dụ 3: không có kiểu đối số nào phụ hợp trong phương thức và mỗi phương thức thay đổi số đối số tương tự nhau. Th này sẽ không xác định được phương thức nào được gọi.
public class OverloadingCalculation3 { void sum(int a, long b) { System.out.println("a method invoked"); } void sum(long a, int b) { System.out.println("b method invoked"); } public static void main(String args[]) { OverloadingCalculation3 obj = new OverloadingCalculation3(); // không xác định được phương thức nào được gọi obj.sum(20, 20); } }
Kết quả:
Compile Time Error