内部类

1. 概念和使用场景

如果一个事物内部包含另一个事物,那么这就是一个类内部包含另一个类。因此,内部类就是定义在一个类或方法内部的类。促使我们使用内部类的原因有两个:

  • 内部类可以对同一个包中的其他类隐藏,即如果我们希望一个类只能被某一个具体的类使用,那么就可以将其定义在哪个类内部作为内部类使用。
  • 内部类方法可以访问定义这个类的作用域中的数据,包括原本私有的数据。如果一个普通类想访问某个类的私有成员属性,那么只能通过类的getter()setter()访问,而内部类可以直接访问它外围类的私有成员属性

2. 分类

2.1 成员内部类

2.1.1 定义格式
  1. // 外部类
  2. 修饰符 class 类名称{
  3. // 内部类
  4. 修饰符 class 类名称{
  5. ...
  6. }
  7. ...
  8. }

只有类为内部类,它的修饰符才有可能是private

2.1.2 数据字段访问

内部类可以访问自身的数据字段,也可以访问创建它的外部类对象的数据字段,即内部类可以直接通过属性名方法访问外部类的数据字段(这种情况建立在内部类中没有和其重名的变量存在),如

  1. public class Body {
  2. private String name = "body";
  3. private String state = "healthy";
  4. public String getName() {
  5. return name;
  6. }
  7. public void setName(String name) {
  8. this.name = name;
  9. }
  10. public String getState() {
  11. return state;
  12. }
  13. public void setState(String state) {
  14. this.state = state;
  15. }
  16. public void methodBody(){
  17. System.out.println("outer class...");
  18. new Heart().beat();
  19. }
  20. public class Heart{
  21. public void beat(){
  22. System.out.println(state);
  23. System.out.println("beat...");
  24. }
  25. }
  26. }
  27. public class InterClassTest {
  28. public static void main(String[] args) {
  29. Body body = new Body();
  30. body.methodBody(); // outer class... healthy beat...
  31. }
  32. }

在内部类Heart的beat()可直接使用,而不必通过getState()使用.
内部类对外部类数据字段的访问.png

外部用内部,需要内部类对象。外部类使用内部类对象有两种方式:

  • 间接方式:在外部类的方法中使用内部类,然后main()中只调用外部类方法,如上所示
  • 直接方式:使用代码格式为:
    1. 外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称();


例如直接在main()直接创建内部类对象,通过内部类对象的方法直接方法外部类数据字段

  1. class Body {
  2. private String name = "body";
  3. public String getName() {
  4. return name;
  5. }
  6. public void setName(String name) {
  7. this.name = name;
  8. }
  9. public void methodBody(){
  10. System.out.println("outer class...");
  11. // 间接调用内部类
  12. new Heart().beat();
  13. }
  14. //成员内部类
  15. public class Heart{
  16. public void beat(){
  17. System.out.println("beat...");
  18. }
  19. }
  20. }
  21. public class InterClassTest {
  22. public static void main(String[] args) {
  23. // 直接调用内部类
  24. Body.Heart heart = new Body().new Heart();
  25. heart.beat(); // beat...
  26. }
  27. }

2.2 局部内部类

如果一个类是定义在一个方法内部,那么它就是一个局部内部类;只有当前所属方法才能使用它,出了这个方法的范围就不能再用。格式:

  1. 修饰符 class 外部类名称{
  2. 修饰符 返回值类型 外部类名称(参数列表){
  3. class 局部内部类名称{
  4. ...
  5. }
  6. ...
  7. }
  8. ...
  9. }

声明局部内部类时不能有访问控制符(即public或private)

代码示例:

  1. class Person{
  2. public void show(){
  3. class Teacher{
  4. public void show(){
  5. System.out.println("teacher...");
  6. }
  7. }
  8. Teacher t = new Teacher();
  9. t.show();
  10. }
  11. }
  12. public class LocalInterClass {
  13. public static void main(String[] args) {
  14. Person p = new Person();
  15. p.show(); // teacher...
  16. }
  17. }

局部内部类如果希望访问所在方法的局部变量,那么这个变量必须是有效final的

2.3 匿名内部类

如果接口的实现类(或父类的子类)只需要使用唯一的一次,则可以省略该类的定义,而改为使用匿名内部类。格式:

  1. 接口名称 对象名 = new 接口名称(){
  2. //覆写所有抽象方法
  3. };

注意事项:对于上面创建匿名内部类的语句来说:

  • new代表创建对象的动作
  • 接口名称是匿名内部类需要实现那个接口
  • {…}是匿名内部类的内容

匿名内部类和匿名对象:

  • 匿名内部类在创建对象的时候,只能使用唯一的一次如果希望希望多次创建多次,而且类的内容完全一样,则只能使用接口的单独实现类
  • 匿名内部类是省略了实现类/子类名称,但匿名对象是省略了对象名称

2.4 静态内部类

内部类的使用是希望只有创建它的外部类使用,对包内其他类具有不可见性。如果同时又希望使用内部类只是将其隐藏在外部类内部,并不需要内部类有外部类对象的一个引用,就可以将内部类类型声明为static,此时内部类就是静态内部类。

例如,创建一个类ArrayGetMaxMin实现获取一次性获取数组中的最大值和最小值,为了保存最值结果,在ArrayGetMaxMin中创建内部类pair实现结果保存。由于pair并不需要有ArrayGetMaxMin对象的引用,因此最好定义为一个静态内部类。

  1. import java.util.Arrays;
  2. import java.util.Random;
  3. public class ArrayGetMaxMin {
  4. public static Pair getMaxMin(int[] arr, int bound){
  5. int min = arr[0];
  6. int max = arr[0];
  7. for(int i = 1; i < arr.length; i++){
  8. if (arr[i]< min){
  9. min = arr[i];
  10. }
  11. if (arr[i] > max){
  12. max = arr[i];
  13. }
  14. }
  15. return new Pair(min, max);
  16. }
  17. public static class Pair{
  18. private int first;
  19. private int second;
  20. public Pair(int first, int second) {
  21. this.first = first;
  22. this.second = second;
  23. }
  24. public int getFirst() {
  25. return first;
  26. }
  27. public int getSecond() {
  28. return second;
  29. }
  30. }
  31. public static void main(String[] args) {
  32. int[] array = new int[10];
  33. int bound = 10;
  34. for (int i = 0; i < array.length; i++) {
  35. array[i] = new Random().nextInt(bound);
  36. }
  37. System.out.println(Arrays.toString(array));
  38. Pair maxMin = ArrayGetMaxMin.getMaxMin(array, bound);
  39. System.out.println("min number is: " + maxMin.getFirst() + " and ma x number is: " + maxMin.getSecond());
  40. }
  41. }

2.5 变量重名问题

通过关键字指定访问的是哪一个变量,如果是当前方法的变量则直接访问;如果是当前类成员变量,需使用this关键字指定访问;如果是外部类的成员变量,需使用外部类名称.this.成员变量名称访问。

  1. class Body {
  2. private String name = "body";
  3. //成员内部类
  4. public class Heart{
  5. String name = "HEART";
  6. public void show(){
  7. String name = "heart";
  8. // 解决重名变量问题
  9. System.out.println(name);
  10. System.out.println(this.name);
  11. System.out.println(Body.this.name);
  12. }
  13. }
  14. }
  15. public class InterClassTest {
  16. public static void main(String[] args) {
  17. Body.Heart heart = new Body().new Heart();
  18. heart.show();
  19. /*
  20. * heart
  21. * HEART
  22. * body
  23. */
  24. }
  25. }

2.6 访问控制修饰符问题

包含有外部类、内部类的访问控制修饰符的使用:

  • 外部类:public、(default)
  • 成员内部类::public / protected / (default) / private
  • 局部内部类:什么都不能写