成员内部类
分类:
1.成员内部类
2.局部内部类(包含匿名内部类)
注意:
1.内部类可以直接访问外部类,外部类需要通过内部类对象来访问内部类。
外部类访问内部类:
1.间接:通过外部类对象,调用外部类的方法,在外部类的方法里面间接使用内部类
2.直接:按照公式 外.内 ——如Body.Heart heart=new Body().new Heart(); heart.beat();
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("外部类的方法");
// Heart heart=new Heart();
// heart.beat();
new Heart().beat();
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
public class Demo01InnerClass {
public static void main(String[] args) {
//通过外部类对象,调用外部类的方法,在外部类的方法里面间接使用内部类Heart
Body body=new Body();
body.methodBody();
System.out.println("-----------------------");
//按照公式 外.内
Body.Heart heart=new Body().new Heart();
heart.beat();
}
}
内部类的同名变量访问
public class Outer {
int num=10;//外部类的成员变量
public class Inner{
int num=20;//内部类的成员变量
public void methodInner(){
int num=30;//内部类方法的局部变量
System.out.println(num);//局部变量 30
System.out.println(this.num);// 内部类的成员变量 20
System.out.println(Outer.this.num);//外部类的成员变量 10
}
}
}
public class Demo02InnerClass {
public static void main(String[] args) {
Outer.Inner obj=new Outer().new Inner();
obj.methodInner();
}
}
运行结果
局部内部类
如果一个类是定义在一个方法内部的,那么这就是一个局部内部类。
注意:
1.只有当前所属的方法才能使用它,出了这个方法外面就不能用了。
2.局部内部类不加修饰符
public class Outer {
public void methodOuter(){
class Inner{//局部内部类
int num=10;
public void methodInner(){
System.out.println(num);//10
}
}
Inner inner=new Inner();
inner.methodInner();
}
}
public class DemoMain {
public static void main(String[] args) {
Outer obj=new Outer();
obj.methodOuter();
}
}
类的权限修饰符
public > protected > (default) > private
定义一个类的时候,权限修饰符规则:
1.外部类:public /(default)
2.成员内部类: public / protected / (default) / private
3.局部内部类:什么都不能写
局部内部类的final问题
注意:
1.局部内部类,如果希望访问所在方法的局部变量,那么这个局部变量必须是【有效final的】。
2.从Java8+开始,只要局部变量事实不变,那么final关键字可以省略。
原因:
1.new出来的对象在堆内存当中。
2.但是局部变量是跟着方法走的,在栈内存当中。
3.方法运行结束之后,立刻出栈,局部变量立刻消失。
4.但是new出来的对象会在堆内存当中持续存在,直到垃圾回收才消失。
5.因为new出来的局部内部类对象要使用,外部类方法中的局部变量,但是局部变量先死了。
6.所以要把局部变量拷贝一份,但是拷贝就要保证局部变量不变,如果改变,那么拷贝过来的数据就不是最新数据,无效。所以要求外部类的方法中局部变量不能变。
public class MyOuter {
public void methodOuter(){
int num=10;//不能改变
class MyInner{
public void methodInner(){
System.out.println(num);
}
}
}
}
匿名内部类
如果接口的实现类(或者是父类的子类) 只需要使用唯一的一次,那么这种情况下就可以省略掉该类的定义,而改为使用【匿名内部类】
对new 接口名称( ) {. . .} 进行格式解析 :
1.new 代表创建对象动作
2.接口名称就是匿名内部类需要实现那个接口
3.{. . . }这才是匿名内部类的内容
注意:
1.匿名内部类,在创建对象时,只能使用唯一一次。如果希望多次创建对象,而且类的内容一样的话,那么就必须使用单独定义的实现类。
2.匿名对象,在调用方法的时候只能调用唯一一次。如果希望同一个对象调用多次方法,那么必须给对象起个名。
3.匿名内部内是省略了实现类/子类,但是匿名对象是省略了对象名称
强调:匿名内部类和匿名对象不是一回事。
public interface MyInterface {
void method1();
void method2();
}
public class MyInterfaceImpl implements MyInterface {
@Override
public void method1() {
System.out.println("实现类覆盖重写了方法!1111");
}
@Override
public void method2() {
System.out.println("实现类覆盖重写了方法!2222");
}
}
public class DemoMain {
public static void main(String[] args) {
// MyInterface obj=new MyInterfaceImpl();
// obj.method();
// MyInterface inter=new MyInterface();//错误写法
//使用匿名内部类
MyInterface objA=new MyInterface() {
@Override
public void method1() {
System.out.println("匿名内部类实现了方法1111 A");
}
@Override
public void method2() {
System.out.println("匿名内部类实现了方法2222 A");
}
};
objA.method1();
objA.method2();
//因为一个匿名内部类,无法创建第二个对象,因此需要再写一遍匿名内部类
MyInterface objB=new MyInterface() {
@Override
public void method1() {
System.out.println("匿名内部类实现了方法1111 B");
}
@Override
public void method2() {
System.out.println("匿名内部类实现了方法2222 B");
}
};
objB.method1();
objB.method2();
System.out.println("--------------------------------------------");
//使用匿名内部类,而且省略了对象名称,也是匿名对象
new MyInterface() {
@Override
public void method1() {
System.out.println("匿名内部类实现了方法1111 B");
}
@Override
public void method2() {
System.out.println("匿名内部类实现了方法2222 B");
}
}.method1();
//因为匿名对象无法调用第二次方法,所以需要再创建一个匿名内部类的匿名对象
new MyInterface() {
@Override
public void method1() {
System.out.println("匿名内部类实现了方法1111 B");
}
@Override
public void method2() {
System.out.println("匿名内部类实现了方法2222 B");
}
}.method2();
}
}