java内部类
作者:互联网
1、介绍
package level2.interClass;
/**
* 1、介绍: 一个类的内部类 又完整的嵌套了另一个类结构,被嵌套的类称为内部类(inner class)
* 嵌套其他类的类称为外部类,是我们类的第五大成员(属性、方法、构造器、代码块、内部类),
* 内部类的最大特点就是可以直接访问私有属性,并且可以体现类与类之间的包含关系
* 2、语法:
* class outer{ // 外部类
* class inner{ // 内部类
* }
* }
* class other{ // 外部其他类
* }
* 4、内部类的分类:
* 1) 定义在外部类的局部位置上(比如方法内):
* a. 局部内部类(有类名)
* b. 匿名内部类(没有类名,是重点)
* 2) 定义在外部类的成员位置上:
* a. 成员内部类(没有static修饰)
* b. 静态内部类(使用static修饰)
*/
public class MyInterClass {
public static void main(String[] args) {
}
}
class Outer{
private int n1 = 100;
public Outer(int n1){
this.n1 = n1;
}
public void m1(){
System.out.println("m1()");
}
{
System.out.println("代码块...");
}
class Inner{ //内部类
}
}
2、局部内部类
package level2.interClass;
/**
* 1、局部内部类:是定义在外部类的局部位置,比如方法中(通常),且有类名
* 1) 可以直接访问外部类所有成员,包含私有的
* 2) 不能添加访问修饰符,因为它的地位就是一个局部变量,局部变量是不能使用修饰符的。
* 但是可以使用final修饰,因为局部变量也能使用final
* 3) 作用域:仅仅在定义它的方法或代码块中 (本质仍然是个类)
* 4) 局部内部类--访问-->外部类的成员-->[直接访问]
* 5) 外部类--访问-->局部内部类的成员-->[创建对象再访问,必须在作用域内]
* 6) 外部其他类--不能访问-->局部内部类
* 7) 如果外部类和局部内部类重名,默认遵循就近原则,如果想访问外部类的成员,则可以使用
* (外部类名.this.成员)去访问
*
*/
public class LocalInnerClass {
public static void main(String[] args) {
MyOuter myOuter = new MyOuter();
myOuter.m1();
}
}
class MyOuter{
private int n1=210;
public void m2(){
System.out.println("MyOuter m2()...");
}
public void m1(){
// 不能添加访问修饰符,但是可以使用final修饰
// 作用域仅仅在m1方法体内
final class MyInner{ // 本质仍是一个类
private int n1 = 900; // 内部类中含有同名的n1
public void f1(){
// 可以直接访问外部类的成员,包括私有属性
// 若内部类中含有同名成员,一般遵循就近原则,若指定访问外部类成员,需要 外部类名.this.成员:
// MyOuter.this 本质是外部类的对象,谁调用f1(),MyOuter.this就是谁
System.out.println("MyOuter n1=" + MyOuter.this.n1);
// 如无显式指定,则遵循就近原则
System.out.println("MyInner n1="+n1);
m2();
}
}
// 外部类访问内部类的成员,需要创建对象再访问
MyInner myInner = new MyInner();
myInner.f1();
// 若MyInner是public,则可以继承
// class MyInner01 extends MyInner{}
}
{
// 也可以在代码块中
class MyInner02{}
}
}
3、匿名内部类
package level2.interClass;
/**
* 1、介绍:匿名内部类(重要)是定义在外部类的局部位置(方法/代码块),且没有类名
* (1)本质还是一个类 (2)内部类 (3)该类没有名字 (4)同时还是一个对象
* 2、基本语法:
* class 类或接口(参数列表){
* 类体
* };
* 3、说明:
* 1) 可以使用对象调用,也可以直接调用
* 3) 作用域:仅仅在定义它的方法或代码块中 (本质仍然是个类)
* 4) 匿名内部类--访问-->外部类的成员-->[直接访问]
* 5) 外部类--访问-->匿名内部类的成员-->[创建对象再访问,必须在作用域内]
* 6) 外部其他类--不能访问-->匿名内部类
* 7) 如果外部类和匿名内部类重名,默认遵循就近原则,如果想访问外部类的成员,则可以使用
* (外部类名.this.成员)去访问
*/
public class AnoymousInnerClass {
public static void main(String[] args) {
}
}
class Outer01{ //外部类
private int n1 = 10;
public void method() {
// 基于接口的匿名内部类
// 想使用IA接口中的方法,但是不想定义类和实例化对象
// 1)tiger的编译类型是 IA,运行类型是 匿名内部类
// 2)匿名内部类,代码在底层运行时,实际上会创建一个类去实现接口,且会分配一个类名: Outer01$1 (外部类+$i)
// 3)jdk底层在创建了匿名内部类Outer01$1,立即就创建了对象实例,并且返回地址
// 4)匿名内部类使用一次,就不能再使用,但是实例化出来的对象仍可以继续使用
// 5)匿名内部类是Outer01$1,tiger是对象
/* 6) 等价于
class Outer01$1 implement tiger {}
*/
IA tiger = new IA() {
public void cry() {
System.out.println("老虎叫...");
}
};
System.out.println("tiger的运行类型 = " + tiger.getClass());
tiger.cry(); // 调用
// 基于类的匿名内部类
// 1) tiger的编译类型是 Father,运行类型是 Outer01$2
/* 2) 等价于:
class Outer01$2 extends Father {}
*/
// 3) 同时返回一个对象
// 注意:参数列表也会传递给构造器
Father father = new Father("tom") {
@Override
public void test() {
System.out.println("匿名内部类重写了test方法");
}
};
System.out.println("father的运行类型 = " + father.getClass());
father.test();
// 注意: 以下形式 不带{} 是实例化对象
Father fa = new Father("tom");
}
}
class Outer02{
private int n1=100;
public void f1(){
// 第一种调用方式:对象调用
Person p = new Person(){
@Override
public void hi() {
System.out.println(" Outer02 中的 第1个hi");
}
};
p.hi();
// 第二种调用方式: 直接调用(可传参)
new Person(){
@Override
public void hi() {
System.out.println(" Outer02 中的 第2个hi");
}
}.hi();
}
}
class Person{
public void hi(){
System.out.println(" Person hi() ");
}
}
interface IA{ // 接口
public void cry();
}
class Father {
public Father(String name) { // 构造器
}
public void test() {
}
}
abstract class Animal{
abstract public void eat();
}
3.1、匿名内部类练习01
package level2.interClass;
/**
* 使用场景
* 1、将匿名内部类 作为实参传递
*
*/
public class InnerClassExercise {
public static void main(String[] args) {
// 将匿名内部类 作为参数传递
f1(new AA() {
@Override
public void show() {
System.out.println("f1() 中的show");
}
});
}
// 静态方法 形参是接口类型
public static void f1(AA aa){
aa.show();
}
}
interface AA{
void show();
}
3.1、匿名内部类练习02
package level2.interClass;
public class InnerClassExercise01 {
public static void main(String[] args) {
//
new CellPhone().alarmclock(new Bell() {
@Override
public void ring() {
System.out.println("懒猪起床了...");
}
});
CellPhone cellPhone = new CellPhone();
cellPhone.alarmclock(new Bell(){
@Override
public void ring() {
System.out.println("开始上课了...");
}
});
cellPhone.alarmclock(new Bell(){
@Override
public void ring() {
System.out.println("已经下课了...");
}
});
}
}
interface Bell{
public void ring();
}
class CellPhone{
public void alarmclock(Bell bell){
bell.ring();
}
}
标签:部类,java,内部,--,void,public,class 来源: https://blog.csdn.net/weixin_41781946/article/details/122814589