java中内部类分为两种:
1.成员内部类 (在成员方法外面定义)
2.局部内部类(其中包括匿名内部类) (在成员方法里面定义,只能在该方法内使用)

成员内部类:内用外,随意访问;外用内,需要内部类对象。
如何使用成员内部类?有两种方式:
1.间接方式:在外部类的方法当中,使用内部类;然后main只是调用外部类的方法。
2.直接方式,公式:
【外部类名称.内部类名称 对象名 = new 外部类名称().new 内部类名称();】

public class Body { // 外部类
    public class Heart { // 成员内部类    
        public void beat() {   // 内部类的方法
            System.out.println("心脏跳动:蹦蹦蹦!");
            System.out.println("我叫:" + name); // 正确写法!
        }
    }   
    private String name; // 外部类的成员变量    
    public void methodBody() {  // 外部类的方法
        System.out.println("外部类的方法");
        new Heart().beat();
    }
}

成员内部类的同名变量访问:
如果出现重名现象,格式是:外部类名称.this.外部类成员变量名

// 如果出现了重名现象,那么格式是:外部类名称.this.外部类成员变量名
public class Outer {
    int num = 10; // 外部类的成员变量
    public class Inner /*extends Object*/ {
        int num = 20; // 内部类的成员变量
        public void methodInner() {
            int num = 30; // 内部类方法的局部变量
            System.out.println(num); // 局部变量,就近原则
            System.out.println(this.num); // 内部类的成员变量
            System.out.println(Outer.this.num); // 外部类的成员变量
        }
    }
}

局部内部类:
注意:如果局部内部类想访问所在方法的局部变量,那么这个局部变量必须是【有效final的】,从java8开始,只要保证该局部变量事实上没变,final关键字可以省略。
原因是声明周期不一样:
1.new出来的对象在堆内存当中。
2.局部变量是跟着方法走的,在栈内存当中。
3.方法运行结束之后,立刻出栈,局部变量就会立刻消失。
4.但是new出来的对象会在堆当中持续存在,直到垃圾回收消失。

如果接口的实现类(或者是父类的子类)只需要使用唯一的一次,
那么这种情况下就可以省略掉该类的定义,而改为使用【匿名内部类】。

匿名内部类的定义格式:
接口名称 对象名 = new 接口名称() {
// 覆盖重写所有抽象方法
};

另外还要注意几点问题:
1.匿名内部类,在【创建对象】的时候,只能使用唯一一次。
如果希望多次创建对象,而且类的内容一样的话,那么就需要使用单独定义的实现类了。
2.匿名对象,在【调用方法】的时候,只能调用唯一一次。
如果希望同一个对象,调用多次方法,那么必须给对象起个名字。
3.匿名内部类是省略了【实现类/子类名称】,但是匿名对象是省略了【对象名称】
强调:匿名内部类和匿名对象不是一回事!!!