在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类。下面就先来了解一下这四种内部类的用法。
1.成员内部类
public class Circle {
private double radius=0.0;
public static int count=0;
public Circle(double radius){
this.radius=radius;
}
class Draw{
public void drawShape(){
System.out.println("[radius]="+radius);
System.out.println("[count]="+count);
System.out.println("内部类的drawShape方法");
}
}
public static void main(String[] args){
Circle circle=new Circle(0.1);
Circle.Draw draw=circle.new Draw();
draw.drawShape();
}
}
类Draw像是类Circle的一个成员,Circle称为外部类。成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。
不过要注意的是,当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:
外部类.this.成员变量
外部类.this.成员方法
public class Circle {
private double radius=0.0;
public int count=0;
public Circle(double radius){
this.radius=radius;
}
public void print(){
System.out.println("Circle print");
}
class Draw{
private double radius=2.0;
public int count=2;
public void drawShape(){
System.out.println("内部类成员变量的值[radius]="+radius);
System.out.println("内部类成员变量的值[count]="+count);
System.out.println("外部类成员变量的值[radius]="+Circle.this.radius);
System.out.println("外部类成员变量的值[count]="+Circle.this.count);
System.out.println("内部类的drawShape方法");
}
}
public static void main(String[] args){
Circle circle=new Circle(0.1);
Circle.Draw draw=circle.new Draw();
draw.drawShape();
}
}
虽然成员内部类可以无条件地访问外部类的成员,而外部类想访问成员内部类的成员却不是这么随心所欲了。在外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问:
public class Circle {
private double radius=0.0;
public Circle(double radius){
this.radius=radius;
getDrawInstance().drawShape();
}
private Draw getDrawInstance(){
return new Draw();
}
class Draw{
public void drawShape(){
System.out.println("[radius]="+radius);
System.out.println("内部类的drawShape方法");
}
}
public static void main(String[] args){
Circle circle=new Circle(0.0);
}
}
内部类可以拥有private访问权限、protected访问权限、public访问权限及包访问权限。比如上面的例子,如果成员内部类Inner用private修饰,则只能在外部类的内部访问,如果用public修饰,则任何地方都能访问;如果用protected修饰,则只能在同一个包下或者继承外部类的情况下访问;如果是默认访问权限,则只能在同一个包下访问。这一点和外部类有一点不一样,外部类只能被public和包访问两种权限修饰。我个人是这么理解的,由于成员内部类看起来像是外部类的一个成员,所以可以像类的成员一样拥有多种权限修饰。
2.局部内部类
局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。
public class Person {
public Person(){
}
class Man{
public Man(){
}
}
public Person getWoman(){
class Woman extends Person{
int age=0;
}
return new Woman();
}
}
局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。
3.匿名内部类
匿名内部类是唯一一种没有构造器的类。正因为其没有构造器,所以匿名内部类的使用范围非常有限,大部分匿名内部类用于接口回调。匿名内部类在编译的时候由系统自动起名为XXX$1.class。一般来说,匿名内部类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的实现或是重写。
public class MyThread {
public static void main(String[] args){
new Thread(){
public void run() {
System.out.println();
};
}.start();
}
}
4.静态内部类
静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法,这点很好理解,因为在没有外部类的对象的情况下,可以创建静态内部类的对象,如果允许访问外部类的非static成员就会产生矛盾,因为外部类的非static成员必须依附于具体的对象。
public class Test {
public static void main(String[] args) {
Outter.Inner inner = new Outter.Inner();
}
}
class Outter {
public Outter() {
}
static class Inner {
public Inner() {
}
}
}
两道笔试试题:
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;
}
}
}
答案
Test t=new Test();
Test.Bean1 bean1=t.new Bean1();
bean1.i++;
Test.Bean2 bean2=new Test.Bean2();
bean2.j++;
Bean bean=t.new Bean();
Bean.Bean3 bean3=bean.new Bean3();
public class Test {
public static void main(String[] args) {
Test test=new Test();
Outter outter = test.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