转载请注明出处:http://blog.csdn.net/lhy_ycu/article/details/45372395
一、前言
内部类也称寄生类,就是把一个类放在类里面(即内部类的上一级程序单元是类)定义,将其作为外部类的成员。内部类主要用几种定义形式:静态(static)内部类,非静态内部类,匿名内部类(也就是没有名字的寄生类)。内部类的好处就是内部类可以直接外部类的(包括私有)成员,反之不能。下面我们通过一些实例来详细讲解一下Java中内部类的使用及几种定义形式的相互调用。
二、实例说明
(1)匿名内部类:当程序创建匿名内部类时,会立即创建匿名内部类(实现类)的实例。
interface IBreathe {
void breathe();
}
/**
* 匿名内部类的使用,定义形式如下:
*
* new 接口() | 父类构造器<参数>){
* // 类体部分...
* };
*
* @author [*昨日重现*] [email protected]
* @since version 1.0
* @datetime 2015年4月29日 下午10:17:40
*/
public class Anonymous {
// 这里相当于创建了接口IBreathe的匿名实现类,并创建了匿名内部类的实例
// 将实现类的实例赋值给接口变量 ,属于向上转型
IBreathe breathe = new IBreathe() {
// 必须实现接口里的所有抽象方法
@Override
public void breathe() {
System.out.println("呼吸新鲜空气...");
}
};
public static void main(String[] args) {
Anonymous anonymous = new Anonymous();
anonymous.breathe.breathe();
}
}
注意以下几点:
1)只要父类是抽象类或者是一个接口,那么其子类中的方法都可以使用匿名内部类来实现;反过来也就是说使用匿名内部类有个前提:它必须显示地继承一个父类或实现一个接口。
2)匿名内部类必须实现接口或抽象类中所有的抽象方法。
3)匿名内部类适合创建只需一次使用的类。
4)匿名内部类不能有构造器而且程序以后无法再访问它,因为它没有类名。
(2)静态内部类:它属于外部类(宿主类)的静态成员,所以它不能访问外部类的非静态成员(属性、方法)。
/**
* 外部类(宿主类)中包含静态内部类作为外部类的一个静态成员
*
* @author [*昨日重现*] [email protected]
* @since version 1.0
* @datetime 2015年4月29日 下午11:00:51
*/
public class Out {
private static String name = "张三";
private String sex = "男";
static class In {
private static String name = "李四";
public void info() {
// 静态内部类不能访问外部类的非静态成员,所以下面这句会编译不通过
// System.out.println("外部类的sex = " + sex);
System.out.println("in的info方法被调用,name:" + name);
}
}
public static void main(String[] args) {
// 打印结果(编译器遵循就近原则):in的info方法被调用,name:李四
// 这里只需要把外部类当成静态内部类的包就行了
// 所以In前加不加Out.都是一样的
new In().info();
new Out.In().info();
}
}
如果要在外部类的外面访问静态内部类的成员:Out.In in = new Out.In();in.info();当然,静态内部类也可以派生子类:class InSub extends Out.In {},编写和调用方式与普通类一样,这里不再赘述。下面看看重点来看看非静态内部类定义和使用。
(3)非静态内部类:在创建非静态内部类的实例前,必须先创建外部类实例,也就是说非静态内部类必须寄生在外部类的实例(Outer.this)里面。所以在创建非静态内部类的实例前,必须先创建外部类实例。
/**
* 非静态内部类的使用
*
* @author [*昨日重现*] [email protected]
* @since version 1.0
* @datetime 2015年4月29日 下午11:26:06
*/
public class Outer {
int number = 10;
// 定义非静态内部类
class Inner {
int number = 100;
public void info() {
int number = 1000;
System.out.println("Inner内部类的info()方法被调用~");
System.out.println("number变量的值为:" + number);// 方法的局部变量,所以为1000
System.out.println("number变量的值为:" + this.number);// 指向寄生类成员,所以为100
System.out.println("number变量的值为:" + Outer.this.number);// 指向宿主类成员,所以为10
}
}
/**
* 通过外部类的方法来访问寄生类的成员(属性、方法)
*/
public void info() {
System.out.println("Outer外部类的info()方法被调用~");
Inner inner = new Inner();
inner.info();
System.out.println(inner.number);
}
public static void main(String[] args) {
new Outer().info();
}
}
如果在外部类的外面使用内部类,那么在创建非静态内部类的实例前,必须先创建外部类实例。
Outer outer = new Outer();
Outer.Inner in = outer.new Inner();
等价于==》Outer.Inner in2 = new Outer().new Inner();
当然非静态内部类也可以派生子类,如下:
//非静态内部类派生子类
class InnerSub extends Outer.Inner {
public InnerSub() {
// 由于Outer.Inner是非静态内部类,因此必须使用"宿主对象"来调用它的构造器
new Outer().super();
}}
三、总结
1)有static修饰的内部类属于外部类本身,没有static修饰的内部类属于类实例 。
2)记住方法是谁的,就用谁来调用。
3)没有static修饰的内部类,【必须寄生在“外部类”的实例里】;反之则寄生在外部类本身里。
4)静态内部类是寄生在类本身里面的,所以就不需要程序员理会宿主。-把外部类当成静态内部类的包就行了。
5)非静态内部类派生子类:由于子类的构造器必须调用父类构造器一次,因此必须在子类构造器中使用宿主对象来调用它的构造器。
四、结束语
关于Java中的内部类的定义和使用基本内容就这些,后面会陆续更新包括:正则表达式、Java性能优化、学习JVM等等,敬请关注。