![名师讲坛:Java开发实战经典(第2版)](https://wfqqreader-1252317822.image.myqcloud.com/cover/824/26793824/b_26793824.jpg)
5.14 内部类
5.14.1 内部类的基本定义
现在已经知道,在类内部可定义成员变量与方法,有趣的是,在类内部也可以定义另一个类。如果在类Outer的内部再定义一个类Inner,此时类Inner就称为内部类,而类Outer则称为外部类。
内部类可声明成public或private。当内部类声明成public或private时,对其访问的限制与成员变量和成员方法完全相同。下面列出了内部类的定义格式:
【格式5-8 内部类的声明格式】
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P147_124239.jpg?sign=1739284997-BYFUerp2Y3PjlR2RKTcBas2P5NLFKhb8-0-b6311d51c54673bfeacfa3df935f5b4d)
下面就根据以上的格式定义一个内部类。
【例5.69】定义内部类
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P147_124240.jpg?sign=1739284997-8ceH5hJzU5adoeGwJlUtoaqwpc2i6F5G-0-77b58036ab9a187541a34fe9be273bd2)
程序执行结果:
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P148_124242.jpg?sign=1739284997-5pt4uTCrCPA5ObCfObVN4iWB9c7xHREJ-0-0925d88e5fc7663aa080a26b4a3d9bb6)
从上面的程序中可以清楚地发现,Inner类作为Outer类的内部类存在,并且在外部类的fun()方法之中直接实例化内部类的对象并调用方法print(),但是从上面的代码之中可以明显地发觉内部类的存在实际上已经破坏了一个类的基本结构,因为类是由属性及方法组成,所以这是内部类的一个缺点,那么内部类有哪些优点呢?如果现在读者把内部类拿到外面来就能发现内部类的优点,如下面代码。
【例5.70】将内部类拿到外部
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P148_18997.jpg?sign=1739284997-Tb76SvR1BvWXYlM0HQfGmaruJ6OOarrD-0-0c0243f42db7b9136df4827af72dd6c3)
程序执行结果:
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P148_124244.jpg?sign=1739284997-SVg3dUGlvkgnTgOETKTa3K3ZOBOyEUru-0-776728d8f0b70797a088b53bd070f9d7)
以上的程序完成了内部类同样的功能,但是代码明显比之前的更加复杂,所以内部类的唯一好处就是可以方便地访问外部类中的私有属性。同时,外部类也可以方便地访问内部类中的私有属性,这样就可以避免封装所带来的操作结构复杂的问题。
5.14.2 使用static定义内部类
使用static可以声明属性或方法,而使用static也可以声明内部类,用static声明的内部类则变成外部类,但是用static声明的内部类不能访问非static的外部类属性。
【例5.71】使用static声明内部类
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P149_124245.jpg?sign=1739284997-lutjIzXi0RVhwyY5xNRvvu6XKneMZ16O-0-001723fa4b31ae7144ca4f4baa509f0f)
程序执行结果:
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P149_124246.jpg?sign=1739284997-U7O7PI91TI0fKY2IA129MPj2lD82VDxO-0-49ec32856810e03687d5e369602a308a)
以上程序中将info属性定义成了static类型,这样程序中就可以通过static声明的内部类直接访问此static属性了,当然,如果此时info属性不是static类型的话,则编译时将出现以下的错误:
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P149_124247.jpg?sign=1739284997-KoOwwybV0VL2vHiXuOBHgP4IOxwXYRwW-0-ef83e870cbc5254307b90ba9c69fcedb)
5.14.3 在外部访问内部类
一个内部类除了可以通过外部类访问,也可以直接在其他类中进行调用,调用的基本格式为:
【格式5-9 在外部访问内部类】
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P149_124248.jpg?sign=1739284997-coNJtnWLw8BfBhSADfRz4daqCXOCc0yB-0-737755f848f189192c2711b7ff54c167)
以上的操作格式中,首先要找到外部类的实例化对象之后,才可以通过外部类的实例化对象去实例化内部类的对象。
提示
观察内部类的class文件。
内部类定义之后,生成的class文件是以“Outer$Inner.class”的形式存在的,在Java中只要是在文件中看见了“$”,则在程序中使用将其替换为“.”。
【例5.72】在外部访问内部类
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P150_124253.jpg?sign=1739284997-PVlB6XQZrXEkrjUKj0oFgZWzbdodh8g9-0-a8f35da3683e0ee1908cfef9008a82a2)
程序执行结果:
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P150_124254.jpg?sign=1739284997-YARtK9uL0DrWWah11FClkgPtawcF8E09-0-78a8854102a41b1790b07843a96dd082)
5.14.4 在方法中定义内部类
理论上,内部类可以定义在程序的任意位置,例如:代码块中、方法中,但是在方法中定义的内部类不能直接访问方法中的参数,如果方法中的参数想被内部类访问,则参数前必须加上final关键字。
【例5.73】在方法中定义内部类
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P150_124255.jpg?sign=1739284997-ArhP9pZ73kvcqkvFudSyTRLa5OXpr75d-0-b9c60c3efd7eb3fa63938fe1848fceed)
程序执行结果:
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P150_124256.jpg?sign=1739284997-Df5J8lCICLaf9HNwQ6wqchXKWhPgAFze-0-1244a2383249b0e8ecffd9f07c7da946)
本程序在fun()方法中定义有Inner的内部类,这样内部类就可以直接进行方法中变量的访问了。
提示
关于JDK 1.8的新支持。
从JDK 1.8开始由于引入了Lamda表达式的原因,所以对于内部类使用方法中的参数也发生了变化。在JDK 1.8以前的所有版本中,如果方法中的参数要想被内部类所访问,则参数前必须加上final关键字。
![](https://epubservercos.yuewen.com/DE8B7B/15253389304122106/epubprivate/OEBPS/Images/Figure-P151_124260.jpg?sign=1739284997-6BCzmuljXEd0kNzXaNxyMQRQtdQNAUVT-0-7ead7b6e97eac73544dd46d1d5bd4fa2)
但是从JDK 1.8开始,此种规则被打破了,即使不加上final也可以正常使用。