Java从入门到精通——第11章 类的高级特性

5.内部类

如果在类中再定义一个类,则将在类中再定义的那个类称为内部类。内部类可以分为成员内部类,局部内部类以及匿名内部类。

5.1成员内部类

1.成员内部类简介
在一个类中使用内部类,可以在内部类中直接存取其所在类的私有成员变量。
成员内部类的语法如下:

public class OuterClass{		//外部类
	private class InnerClass{	//内部类
		//...
	}
}

在内部类中可以随意使用外部类的成员方法以及成员变量,尽管这些类成员被修饰为private。
内部类的实例一定要绑定在外部类的实例上,如果从外部类中初始化一个内部类对象,那么内部类对象就会绑定在外部类对象上。内部类初始化方式与其他类初始化方式相同,都是使用new关键字。
下面看一个实例:在项目中创建OuterClass类,在类中定义innerClass内部类和doit()方法,在主方法中创建OuterClass类的实例对象和doit()方法。

public class OuterClass {
    class innerClass{
        innerClass(){}                      //内部类的构造方法
        public void inf(){}                 //内部类的成员方法
        int y = 0;                          //定义内部类的成员变量
    }

    innerClass in = new innerClass();       //在外部类实例化内部类对象引用
    public void ouf(){
        in.inf();                           //在外部类方法中调用内部类方法
    }
    public innerClass doit(){               //外部类方法,返回值为内部类引用
//      y = 4;                              //外部类不可以直接访问内部类成员变量
        in.y = 4;
        return new innerClass();            //返回内部类引用
    }

    public static void main(String[] args){
        OuterClass out = new OuterClass();
        //内部类的实例化操作必须在外部类或外部类的非静态方法中实现
        OuterClass.innerClass in = out.doit();
        OuterClass.innerClass in2 = out.new innerClass();
    }
}

上例中的外部类创建内部类实例与其他类创建对象引用时相同。内部类可以访问它的外部类成员,但内部类的成员只有在内部类的范围内是可知的,不能被外部类引用,但是可以使用内部类对象引用调用成员变量y。
**注意:**如果在外部类和非静态方法之外实例化内部类对象,需要使用外部类。内部类的形式指定该对象的类型。
在静态方法中为啥不能创建内部类的实例对象

2.内部类向上转型为接口
如果将一个权限修饰符为private的内部类向上转型为其父类对象,或者直接向上转型为一个接口,在程序中就可以完全隐藏内部类的具体实现过程。可以在外部提供一个接口,在接口中声明一个方法。如果在实现该接口的内部类中实现该接口的方法,就可以定义多个内部类以不同的方式实现接口中的同一个方法,而在一般的类中是不能多次实现接口中的同一个方法的。
**实例:**在项目中创建InterfaceInner类,并定义接口OutInterface,使内部类InnerClass实现这个接口,最后使doit()方法返回值类型为接口。代码如下:

interface OutInterface{                             //定义一个接口
    public void f();
}

class OutClass2 {
    //定义一个内部类实现OutInterface接口
    private class InnerClass implements OutInterface{
        InnerClass(String s){                       //内部类构造方法
            System.out.println(s);
        }
        public void f() {                           //实现接口中的f()方法
            System.out.println("访问内部类中的f()方法");
        }
    }

    //定义一个方法,返回类型为OutInterface接口
    public OutInterface doit(){
        return new InnerClass("访问内部类构造方法");
    }
}

public class InterfaceInner {
    public static  void main(String[] args){
        OutClass2 out = new OutClass2();            //实例化一个OuterClass2对象
        OutInterface  outinter = out.doit();        //调用doit()方法,返回一个OutInterface接口
        outinter.f();                               //调用f()方法
    }
}

在这里插入图片描述
从上可以看出,OuterClass2类定义了一个权限修饰为private的内部类,这个内部类实现了OutInterface接口,然后修改doit()方法,使该方法返回一个OutInterface接口。由于内部类InnerClass权限修饰为private,所以除了OuterClass2类可以访问该内部类之外,其他类都不能访问,而可以访问doit()方法。由于该方法返回一个外部接口类型,这个接口可以作为外部使用的接口。它包含了一个f()方法,在继承此接口的内部类中实现了该方法,如果某个类继承了外部类,由于内部的权限不可以向下转型为内部类InnerClass,同时也不能访问f()方法,但是可以访问接口中的f()方法。例如,InterfaceInner类中最后一条语句,接口引用调用f()方法,从执行结果可以看出,这条语句执行的是内部类中的f()方法,很好的对继承该类的子类隐藏了实现细节,仅为编写子类的人留下了一个接口和一个外部类,同时也可以调用f()方法,但是f()方法的具体实现过程却被很好的隐藏了,这就是内部类最基本的用途。

3.使用this关键字或取内部类与外部类的引用
如果在外部类中定义的成员变量与内部类的成员变量名称相同,可以使用this关键字。
实例:在项目中创建TheSameName类,在类中定义成员变量x,再定义一个内部类Inner,在内部中也创建x变量,并在内部类的doit()方法中分别操作两个x变量。关键代码如下:

public class TheSameName {
    private int x;
    private class Inner{
        private int x = 9;
        public void doit(int x){
            x++;                        //调用的是形参
            this.x++;                   //调用的是内部类的变量x
            TheSameName.this.x++;       //调用的是外部类的变量x
        }
    }
}

在类中,如果遇到内部类与外部类的成员变量重名的情况,可以使用this关键字进行处理。例如,在内部类中使用this.x语句可以调用内部类的成员变量x,而使用TheSameName.this.x语句可以调用外部类的成员变量x。
在内存中所有对象均被放置在堆中,方法以及方法中的形参或局部变量放置在栈中。

5.2局部内部类

内部类不仅可以在类中定义,也可以在类的局部位置定义,如在类的方法或任意的作用域中均可定义内部类。
实例:将InnerClass类放在doit()方法的内部。代码如下

interface OutInterface2{                        //定义一个接口
    
}

public class OuterClass3 {
    public OutInterface2 doit(final String x){  //doit()方法参数为final类型
        //在doit()方法中定义一个内部类
        class InnerClass2 implements OutInterface2{
            InnerClass2(String s){
                s = x;
                System.out.println(s);
            }
        }
        return new InnerClass2("doit");
    }
}

从上可以看出,内部类InnerClass2是doit()方法的一部分,并非OuterClass3类中的一部分,所以在doit()方法的外部不能访问该内部类,但是该内部类可以访问当前代码块的常量以及此外部类的所有成员。
上面将doit()方法的参数设置为final类型。如果需要在方法体中使用局部变量,该局部变量需要被设置为final类型的,换句话说,在方法定义的内部类只能访问方法中final类型的局部变量,这是因为在方法中定义的局部变量相当于一个常量,它的生命周期超出方法运行的生命周期,由于该局部变量被设置为final,所以不能在内部类中改变该局部变量的值。

5.3匿名内部类

在doit()方法中将return语句和内部类定义结合合并在一起,下面通过一个实例说明。
实例:在return语句中编写返回值为一个匿名内部类

public class OuterClass4 {
    public OutInterface2 doit(){            //定义doit()方法
        return new OutInterface2() {        //声明匿名内部类
            private int i=0;
            public int getValue(){
                return i;
            }
        };
    }
}

在doit()方法内部首先返回一个OutInterface2的引用,然后在return语句中插入一个定义 内部类的代码,由于这个类没有名称,所以这里将该内部类称为匿名内部类。实质上这种内部类的作用就是创建一个实现于OutInterface2接口匿名类的对象。
匿名类的所有实现代码都需要在大括号之间进行编写。语法如下:

return new A(){
	...//内部类体
};

其中A指类名。
由于匿名内部类没有名称,所以匿名内部类使用默认构造方法来实现OutInterface2对象。在匿名内部类定义结束后,需要加分号标识,这个分号并不是代表定义内部类结束的标识,而是代表创建OutInterface2引用表达式的标识。
Tips:匿名内部类编译以后,会产生以“外部类名$序号"为名称的.class文件,序号以1 ~ n排列,分别代表1 ~ n个匿名内部类。

5.4静态内部类

在内部类前添加修饰符static,这个内部类就变成静态内部类了。一个静态内部类中可以声明static成员,但是在非静态内部类中不可以声明静态成员。静态内部类由一个最大的特点,就是不可以使用外部类的非静态成员,所以静态内部类在开发中比较少见。
可以这样认为,普通的内部类对象隐式地在外部保存了一个引用,指向创建它的外部类对象,但如果内部类被定义为static,就会有更多的限制。静态内部类有以下两个特点:
1.如果创建静态内部类的对象,不需要其外部类的对象。
2.不能从静态内部类的对象中访问非静态外部类的对象。
实例:定义一个静态内部类StaticInnerClass,可以使用如下代码:

public class StaticInnerClass{
	int x = 100;
	static class Inner{
		void doitInner(){
			//System.out.println("外部类" + x);		//调用外部类的成员变量x
		}
	}
}

进行程序测试时,如果在每一个Java文件中都设置一个主方法,将出现很多额外代码,而程序本身并不需要这些方法,为了解决这个问题,可以将主方法写入静态内部类中
实例:在静态内部类中定义主方法。

public class StaticInnerClass{
	int x = 100;
	static class Inner{
		void doitInner(){
			//System.out.println("外部类" + x);
		}
		public static void main(String[] args){
			System.out.println("a");
		}
	}
}

如果编译上面的类,将生成一个名称为StaticInnerClass $ Inner的独立类和一个StaticInnerClass类,只要使用java StaticInnerClass $I nner,就可以运行主方法中的内容,这样完成测试,需要将所有.class文件打包时,只要删除StaticInnerClass $ Inner独立类即可。

5.5内部类的继承

内部类和其他普通一样可以被继承,但是继承内部类比继承普通类复杂,需要设置专门的语法来完成。
实例:在项目中创建OutputInnerClass类,使OutputInnerClass类继承ClassA中的内部类ClassB

public class OutputInnerClass extends ClassA.ClassB{		//继承内部类ClassB
	public OutputInnerClass(ClassA a){
		a.super();
	}
}
class ClassA{
	class ClassB{
	}
}

在某个类继承内部类时,必须硬性给予这个类一个带参数的构造方法,并且该构造方法的参数为需要集成内部类的外部类的引用,同时在构造方法体中使用a.super()语句,这样才为继承提供了必要的对象引用。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值