JAVA-内部类

1. 内部类

1.1 概述

​ 把类定义在另一个类的内部,该类就被称为内部类。

​ 举例:把类B定义在类A中,类B就被称为内部类。

1.2 访问规则

  1. 可以直接访问外部类的成员,包括私有。
  2. 外部类要想访问内部类成员,必须创建对象。
class Outer {
    private int num = 10;
    class Inner {
        public void show() {
            //可直接访问外部类的成员,包括私有
            System.out.println(num);
        }
    }
    
    public void method() {
        //若要访问Inner类中的方法,需创建其对象
        Inner i = new Inner();
        i.show();
    }
}

1.3 分类

  • 成员位置:在成员位置定义的类,被称为成员内部类。

  • 局部位置:在局部位置定义的类,被称为局部内部类。

class Outer {
    private int num = 10;
	// 成员位置
	class Inner {...}
	
    public void method() {
        //局部位置
        class Inner {...}
    }
}

2. 成员内部类

2.1 成员内部类的修饰符

  • public:正常修饰符

  • private:为了数据的安全性

//案例:人有身体,身体内有心脏。即Body类内有私有的成员内部类Heart()。
//只有医生才可调用心脏的operator()方法
class Body {
	private class Heart {
		public void operator() {
			System.out.println("心脏手术");
		}
	}

	public void method(String role) {
		if ("医生".equals(role)) {
			Heart h = new Heart();
			h.operator();
		}
	}
}

//调用方式
Body b =  new Body();
b.method("医生");
  • static:为了访问的方便性

    注意:静态内部类访问的外部类数据必须用静态修饰。

class Outer {
	private int num = 10;
	private static int num2 = 100;

	public static class Inner {
		public void show() {
            // 静态内部类不可访问外部类的非静态成员变量
			// System.out.println(num);
			System.out.println(num2);
		}
	}
}

2.2 成员内部类的访问

2.2.1 成员内部类不是静态的

​ 外部类名.内部类名 对象名 = new 外部类名.new 内部类名();

​ 如:Outer.Inner oi = new Outer().new Inner();

class Outer {
	private int num = 10;

	// 成员内部类不是静态的
	class Inner {
		public void show() {
			System.out.println(num);
		}
	}
}

class Demo {
	public static void main(String[] args) {
		// 需求:我要访问非静态Inner类的show()方法
		// 格式:外部类名.内部类名 对象名 = 外部类对象.内部类对象;
		Outer.Inner oi = new Outer().new Inner();
		oi.show();
	}
}
2.2.2 成员内部类是静态的

​ 外部类名.内部类名 对象名 = new 外部类名.内部类名();

​ 如:Outer.Inner oi = new Outer.Inner();

class Outer {
	private int num = 10;
	private static int num2 = 100;
	// 成员内部类是静态的
	public static class Inner {
		public void show1() {
			System.out.println(num2);
		}
		
		public static void show2() {
			System.out.println(num2);
		}
	}

}

class InnerClassDemo3 {
	public static void main(String[] args) {
		// 需求:我要访问静态Inner类的show()、show2()方法
		// 格式:外部类名.内部类名 对象名 = new 外部类名.内部类名();
		Outer.Inner oi = new Outer.Inner();
		oi.show1();
		oi.show2();// 静态类也可直接对象调用
		// show2()的另一种调用方式,即通过类名调用。
		Outer.Inner.show2();
	}
}

2.3 成员内部类的应用面试题

  • 要求:填入代码,分别输出30,20,10。

​ 注意:

​ 1:内部类和外部类没有继承关系。

​ 2:通过外部类名限定this对象。 如:Outer.this

class Outer {
	public int num = 10;

	class Inner {
		public int num = 20;

		public void show() {
			int num = 30;
			System.out.println("请填入代码,输出 30");
			System.out.println("请填入代码,输出 20");
			System.out.println("请填入代码,输出 10");
		}
	}
}

程序如下:

class Outer {
	public int num = 10;

	class Inner {
		public int num = 20;
	
		public void show() {
			int num = 30;
			System.out.println(num); // 输出30
			System.out.println(this.num); // 输出20
			System.out.println(new Outer().num); // 输出10
			System.out.println(Outer.this.num); // 输出10
		}
	}
}

3. 局部内部类

3.1 局部内部类的访问

  • 局部内部类可以直接访问外部类的成员。

  • 在局部位置可以创建内部类对象,通过对象调用内部类方法使用局部内部类功能。

class Outer {
	private int num = 10;

	public void method() {
		class Inner {
			public void show() {
				// 局部内部类可直接访问内部类成员变量
				System.out.println(num);
			}
		}
		// 在局部位置,可以创建内部类对象,通过对象调用内部类方法,来使用局部内部类功能
		Inner i = new Inner();
		i.show();
	}

}

class Test {
	public static void main(String[] args) {
		Outer o = new Outer();
		o.method();
	}
}

3.2 局部内部类访问局部变量的注意事项

局部内部类访问局部变量,局部变量必须用final修饰

原因:局部变量会随着方法的调用完毕而消失,这个时候,局部对象并没有立马从堆内存中消失,还要使用那个变量。为了让数据还能继续被使用,就用fianl修饰,这样,在堆内存里面存储的其实是一个常量值。

class Outer {
	public void method() {
        int num = 10;
		final int num2 = 20;
		class Inner {
			public void show() {
				// 从内部类中访问本地变量num,报错。需要被声明为final类型
                
                //声明为final后,num2是常量,内存中此处已不是num2,而直接替换为20
				System.out.println(num);
			}
		}
		Inner i = new Inner();
		i.show();
	}
}

如果是变量20,那么是随着方法调用而存在,随方法调用结束而消失。

但是创建对象是在堆内存,而不是方法调用的栈内存。

method()方法调用结束之后,对象 i 并没有消失,所以必须是常量,所以必须用final修饰。

而因为是常量,所以输出语句:System.out.println(num2)中的num2已经自动替换成了20,而不再是num2。

3. 匿名内部类

​ 是局部内部类的简化形式

3.1 匿名内部类介绍

3.1.2 前提

​ 存在一个类或者接口,这里的类可以是具体类也可以是抽象类。

3.1.3 格式

​ new 类名或者接口名() {

​ 重写方法;

​ };

不要忘记最后的分号

interface Inter {
	public abstract void show();
}

class Outer {
	public void method() {
	   new Inter() {
		   public void show() {
			   System.out.println("show");
		   }
	   };
	}
}
3.1.3 本质

​ 是继承该类或者实现接口的子类匿名对象

3.2 调用匿名内部类

interface Inter {
	public abstract void show();
}

class Outer {
	public void method() {
		new Inter() {
			public void show() {
				System.out.println("show");
			}
		}.show();
	}
}

而上面这种方法如果show()方法太多,会很繁琐,利用多态调用。

interface Inter {
	public abstract void show();

	public abstract void show2();

}

class Outer {
	public void method() {
		Inter i = new Inter() { // 多态
			public void show() {
				System.out.println("show");
			}

			public void show2() {
				System.out.println("show2");
			}
		}; // 此处别忘了分号
		i.show();
		i.show2();
	}

}

class Test {
	public static void main(String[] args) {
		Outer o = new Outer();
		o.method();
	}
}

3.3 匿名内部类在开发中的使用

​ 在开发的时候,会看到抽象类,或者接口作为参数。而这个时候,我们知道实际需要的是一个子类对象。如果该方法仅仅调用一次,我们就可以使用匿名内部类的格式简化。

interface Person {
	public abstract void study();
}

class PersonDemo {
	public void method(Person p) {
		p.study();
	}
}

class PersonTest {
	public static void main(String[] args) {
		PersonDemo pd = new PersonDemo();
		pd.method(new Person() {
			public void study() {
				System.out.println("好好学习,天天向上");
			}
		});
	}
}

3.4 匿名内部类的面试题

​ 按照要求,补齐代码,要求在控制台输出“HelloWorld”

interface Inter { void show(); }
class Outer { //补齐代码 }
class OuterDemo {
	public static void main(String[] args) {
		  Outer.method().show();
	  }
}

​ 分析:

​ 1:Outer.method()可以看出method()应该是Outer中的一个静态方法。

​ 2:Outer.method().show()可以看出method()方法的返回值是一个对象。由于接口Inter中有一个show()方法,所以我认为method()方法的返回值类型是一个接口,即接口的子类对象。

​ 3:并没有给出接口的子类,所以这里需要一个接口的子类匿名对象。

​ 代码实现:

interface Inter {
	void show(); // 修饰符为public abstract,重写时要注意
}

class Outer {
	// 补齐代码
	public static Inter method() {
		// 子类对象 -- 子类匿名对象
		return new Inter() {
			public void show() {
				System.out.println("HelloWorld");
			}
		};
	}
}

class OuterDemo {
	public static void main(String[] args) {
		Outer.method().show();
	}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值