转自https://www.cnblogs.com/dolphin0520/p/3811445.html 谢谢
一.内部类基础
二.深入理解内部类
三.内部类的使用场景和好处
四.常见的与内部类相关的笔试面试题
若有不正之处,请多谅解并欢迎批评指正。
请尊重作者劳动成果,转载请标明原文链接:
http://www.cnblogs.com/dolphin0520/p/3811445.html
一.内部类基础
在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类。下面就先来了解一下这四种内部类的用法。
1.成员内部类
成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
class
Circle {
double
radius =
0
;
public
Circle(
double
radius) {
this
.radius = radius;
}
class
Draw {
//内部类
public
void
drawSahpe() {
System.out.println(
"drawshape"
);
}
}
}
|
这样看起来,类Draw像是类Circle的一个成员,Circle称为外部类。成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
class
Circle {
private
double
radius =
0
;
public
static
int
count =
1
;
public
Circle(
double
radius) {
this
.radius = radius;
}
class
Draw {
//内部类
public
void
drawSahpe() {
System.out.println(radius);
//外部类的private成员
System.out.println(count);
//外部类的静态成员
}
}
}
|
不过要注意的是,当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:
1
2
|
外部类.
this
.成员变量
外部类.
this
.成员方法
|
虽然成员内部类可以无条件地访问外部类的成员,而外部类想访问成员内部类的成员却不是这么随心所欲了。在外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
class
Circle {
private
double
radius =
0
;
public
Circle(
double
radius) {
this
.radius = radius;
getDrawInstance().drawSahpe();
//必须先创建成员内部类的对象,再进行访问
}
private
Draw getDrawInstance() {
return
new
Draw();
}
class
Draw {
//内部类
public
void
drawSahpe() {
System.out.println(radius);
//外部类的private成员
}
}
}
|
成员内部类是依附外部类而存在的,也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象。创建成员内部类对象的一般方式如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
public
class
Test {
public
static
void
main(String[] args) {
//第一种方式:
Outter outter =
new
Outter();
Outter.Inner inner = outter.
new
Inner();
//必须通过Outter对象来创建
//第二种方式:
Outter.Inner inner1 = outter.getInnerInstance();
}
}
class
Outter {
private
Inner inner =
null
;
public
Outter() {
}
public
Inner getInnerInstance() {
if
(inner ==
null
)
inner =
new
Inner();
return
inner;
}
class
Inner {
public
Inner() {
}
}
}
|
2.局部内部类
局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
class
People{
public
People() {
}
}
class
Man{
public
Man(){
}
public
People getWoman(){
class
Woman
extends
People{
//局部内部类
int
age =
0
;
}
return
new
Woman();
}
}
|
注意,局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。
3.静态内部类
静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法,这点很好理解,因为在没有外部类的对象的情况下,可以创建静态内部类的对象,如果允许访问外部类的非static成员就会产生矛盾,因为外部类的非static成员必须依附于具体的对象。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
public
class
Test {
public
static
void
main(String[] args) {
Outter.Inner inner =
new
Outter.Inner();
}
}
class
Outter {
public
Outter() {
}
static
class
Inner {
public
Inner() {
}
}
}
|
package net.xsoftlab.baike; public class Circle { double radius=0; public static int count =1; public void circle(double radius) { this.radius=radius; } static class Inner{ public void drawshape() { System.out.println(radius);//报错,由于调用了非静态成员 System.out.println(count); } } }
二.深入理解内部类
1.为什么成员内部类可以无条件访问外部类的成员?
在此之前,我们已经讨论过了成员内部类可以无条件访问外部类的成员,那具体究竟是如何实现的呢?下面通过反编译字节码文件看看究竟。事实上,编译器在进行编译的时候,会将成员内部类单独编译成一个字节码文件,下面是Outter.java的代码:
三.内部类的使用场景和好处
为什么在Java中需要内部类?总结一下主要有以下四点:
1.每个内部类都能独立的继承一个接口的实现,所以无论外部类是否已经继承了某个(接口的)实现,对于内部类都没有影响。内部类使得多继承的解决方案变得完整,
2.方便将存在一定逻辑关系的类组织在一起,又可以对外界隐藏。
3.方便编写事件驱动程序
4.方便编写线程代码
个人觉得第一点是最重要的原因之一,内部类的存在使得Java的多继承机制变得更加完善。
四.常见的与内部类相关的笔试面试题
1.根据注释填写(1),(2),(3)处的代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
public
class
Test{
public
static
void
main(String[] args){
// 初始化Bean1
(
1
)
bean1.I++;
// 初始化Bean2
(
2
)
bean2.J++;
//初始化Bean3
(
3
)
bean3.k++;
}
class
Bean1{
public
int
I =
0
;
}
static
class
Bean2{
public
int
J =
0
;
}
}
class
Bean{
class
Bean3{
public
int
k =
0
;
}
}
|
从前面可知,对于成员内部类,必须先产生外部类的实例化对象,才能产生内部类的实例化对象。而静态内部类不用产生外部类的实例化对象即可产生内部类的实例化对象。
创建静态内部类对象的一般形式为: 外部类类名.内部类类名 xxx = new 外部类类名.内部类类名()
创建成员内部类对象的一般形式为: 外部类类名.内部类类名 xxx = 外部类对象名.new 内部类类名()
因此,(1),(2),(3)处的代码分别为:
Test test = new Test(); Test.Bean1 bean1 = test.new Bean1();
Test.Bean2 b2 = new Test.Bean2();
Bean bean = new Bean(); Bean.Bean3 bean3 = bean.new Bean3();
2.下面这段代码的输出结果是什么?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public
class
Test {
public
static
void
main(String[] args) {
Outter outter =
new
Outter();
outter.
new
Inner().print();
}
}
class
Outter
{
private
int
a =
1
;
class
Inner {
private
int
a =
2
;
public
void
print() {
int
a =
3
;
System.out.println(
"局部变量:"
+ a);
System.out.println(
"内部类变量:"
+
this
.a);
System.out.println(
"外部类变量:"
+ Outter.
this
.a);
}
}
}
|
3 2 1
最后补充一点知识:关于成员内部类的继承问题。一般来说,内部类是很少用来作为继承用的。但是当用来继承的话,要注意两点:
1)成员内部类的引用方式必须为 Outter.Inner.
2)构造器中必须有指向外部类对象的引用,并通过这个引用调用super()。这段代码摘自《Java编程思想》
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
class
WithInner {
class
Inner{
}
}
class
InheritInner
extends
WithInner.Inner {
// InheritInner() 是不能通过编译的,一定要加上形参
InheritInner(WithInner wi) {
wi.
super
();
//必须有这句调用
}
public
static
void
main(String[] args) {
WithInner wi =
new
WithInner();
InheritInner obj =
new
InheritInner(wi);
}
}
|