Class cha có Method chứa inner class, method đó sử dụng biến ngoài inner class và trong method, biến đó phải là final hoặc effective final. Môi trường: Java language level 19, JDK 19. Bạn có thể dùng kiến thức này cho các Java language level hoặc JDK rất thấp, tuy nhiên tôi muốn cụ thể hóa cho một phiên bản Java hiện tại.
Ví dụ minh họa, lưu ý rằng nó không thể biên dịch.
public class PrintNumbers { private int length = 4; public void calculate() { final int width = 5; class Calculator { public void multiply() { System.out.println(length * width); } } var calculator = new Calculator(); calculator.multiply(); } public static void main(String[] args) { var printer = new PrintNumbers(); printer.calculate(); } public void processData(){ final int length = 1; int width = 2; int height = 3; class VolumeCalculator{ public int multiply(){ return length * width * height; } } width = 2; }
}
Nguyên nhân gây rà lỗi khoogn thể biên dịch: Tại dòng return length * width * height;
sử dụng biến width
không ở trạng thái final, constructor VolumeCalculator
không thể xác định được width = 3 hay = 2. Có 2 trường hợp cần phân biệt:
- final (1), và
- effective final (2).
Đoạn code trên rơi vào trường hợp (2) đó là effective final. Cách fix có 2 cách --> đưa width
về (1) --> Solution (S1) hoặc (2) --> Solution (S2). Đó là
(S1): Final
public class PrintNumbers { private int length = 4; public void calculate() { final int width = 5; class Calculator { public void multiply() { System.out.println(length * width); } } var calculator = new Calculator(); calculator.multiply(); } public static void main(String[] args) { var printer = new PrintNumbers(); printer.calculate(); } public void processData(){ final int length = 1; final int width = 2; int height = 3; class VolumeCalculator{ public int multiply(){ return length * width * height; } } // width = 2; }
}
S2: Effective final
public class PrintNumbers { private int length = 4; public void calculate() { final int width = 5; class Calculator { public void multiply() { System.out.println(length * width); } } var calculator = new Calculator(); calculator.multiply(); } public static void main(String[] args) { var printer = new PrintNumbers(); printer.calculate(); } public void processData(){ final int length = 1; int width = 2; int height = 3; class VolumeCalculator{ public int multiply(){ return length * width * height; } } // width = 6; }
}
Các bạn thấy đấy, int width = 10
đâu cần final, chương trình vẫn hoạt động được, vì nó không gây ra nhầm lẫn về giá trị biến width được dùng trong method bên trong inner class, cái này gọi là effective final (final một cách ngầm định).
Tài liệu đặc tả ngôn ngữ Java - language level 19: https://docs.oracle.com/javase/specs/jls/se19/html/jls-8.html#d5e12966:~:text=Example 8.1.3-2. Inner Class Declarations
Mã nguồn: https://gist.github.com/donhuvy/f44cbcdf73f88f8e1b60445e64d8dbfc#file-printnumbers-java-L23