1.成员内部类
定义在一个类(外部类)的内部,而且与成员方法、属性平级的类叫成员内部类,…相当于外部类的非静态方法,如果被static修饰,就变成静态内部类了。
public class Person {
private int age;
private String name;
public void run() {
System.out.println("run.......");
}
class User{
private int age;
}
}
注意事项
1.成员内部类中不能存在static关键字,即,不能声明静态属性、静态方法、静态代码块等。【非静态内部类也可以定义静态成员但需要同时有final关键词修饰,静态方法鉴于无法用final修饰,仍必须是在静态内部类 或者非内部类中定义。】
创建成员内部类的实例使用:外部类名.内部类名 实例名 = 外部类实例名.new 内部类构造方法(参数),可以理解为隐式地保存了一个引用,指向创建它的外部类对象。
2、在成员内部类中访问外部类的成员方法和属性时使用:外部类名.this.成员方法/属性。
3、内部类在编译之后生成一个单独的class文件,里面包含该类的定义,所以内部类中定义的方法和变量可以跟父类的方法和变量相同。例如上面定义的三个类的class文件分别是:MyTest.class、Outer.class和Outer$Inner.class三个文件。
4、外部类无法直接访问成员内部类的方法和属性,需要通过内部类的一个实例来访问。
5、与外部类平级的类继承内部类时,其构造方法中需要传入父类的实例对象。且在构造方法的第一句调用“外部类实例名.super(内部类参数)”。
成员内部类里面为什么不能有静态成员和方法?
非静态内部类不能有静态成员!
成员内部类必须先实例化外部类对象然后再实例化成员内部类;
非static的内部类,在外部类加载的时候,并不会加载它,所以它里面不能有静态变量或者静态方法。
1、static类型的属性和方法,在类加载的时候就会存在于内存中。
2、要使用某个类的static属性或者方法,那么这个类必须要加载到jvm中。
基于以上两点,可以看出,如果一个非static的内部类如果具有static的属性或者方法,那么就会出现一种情况:内部类未加载,但是却试图在内存中创建static的属性和方法,这当然是错误的。原因:类还不存在,但却希望操作它的属性和方法。
静态成员和静态方法是随着类的加载而存在的,也就是说内部类的静态属性是随着类加载的,但是内部类的实例,是创建后才存在的,也就是说其静态属性优先存在于他的类实例, 这显然是矛盾的,所以要把内部类设为静态的 这样他们的生命周期就是相同了;
如果内部类没有static的话,就需要实例化内部类才能调用,说明非static的内部类不是自动跟随主类加载的,而是被实例化的时候才会加载。而static的语义,就是主类能直接通过内部类名来访问内部类中的static方法,而非static的内部类又是不会自动加载的,所以这时候内部类也要static,否则会前后冲突。
2.静态内部类
使用static修饰的成员内部类叫静态内部类。
public class Person {
private int age;
private String name;
public void run() {
System.out.println("run.......");
}
static class User{
private int age;
}
}
3.局部内部类
定义在代码块、方法体内、作用域(使用花括号“{}”括起来的一段代码)内的类叫局部内部类。
public class Person {
private int age;
private String name;
public void run() {
System.out.println("run.......");
//局部内部类
class a{
private int gender;
}
}
static class User{
private int age;
}
}
4.匿名内部类
匿名内部类不能是抽象类,因为匿名内部类在定义之后,会立即创建一个实例。
//基于接口
public interface IUserService {
void eat();
}
public class Person {
private int age;
private String name;
public void run(IUserService iUserService) {
System.out.println("run.......");
}
class User{
void test() {
//使用匿名内部类
run(new IUserService() {
@Override
public void eat() {
// TODO Auto-generated method stub
System.out.println();
}
});
}
}
}