一.内部类学习导图
1>.静态内部类:
使用static修饰符来修饰内部类,则这个内部类就属于外部类本身,而不属于外部类的某个对象。因此使用static修饰的内部类被称为静态内部类。
public class StaticInclass {
private static int num;
private int num1;
private static int num2;
/*
* 创建一个静态内部类
* 静态内部类可以包含静态成员,也可以包含非静态成员。
* 根据静态成员不能访问非静态成员的规则,静态内部类不能访问外部类的实例成员,只能访问外部类的类成员。
* 即使是静态内部类的实例方法也不能访问外部类的实例成员,只能访问外部类的静态成员。
*/
static class InClass{//静态内部类访问权限为包访问权限
private int num;//静态内部类中出现了非静态变量,必须将静态内部类实例化后才能获得
private static int num1;//静态变量
public void f() {
System.out.println(num2);
System.out.println(this.num);//获取了内部类的num符合就近原则
System.out.println(StaticInclass.num);//可以获取静态变量
// System.out.println(StaticInclass.num1);非静态方法无法获取外部类的非静态变量
}
public static void f1() {
System.out.println(StaticInclass.num);//可以获取静态变量
// System.out.println(StaticInclass.num1);静态方法同样无法获取外部类的非静态变量
}
public int getNum() {
return num;
}
}
public void f() {
System.out.println(InClass.num1);//内部类的静态成员变量
System.out.println(new InClass().num);//将静态内部类实例化
}
public static void main(String[] args) {
/*
* 外部类对静态内部类调用方式
*/
StaticInclass s=new StaticInclass(); }
}
注意点:
1.静态内部类允许拥有非静态变量,但是在获得静态内部类的非静态变量,必须将静态内部类实例化。
System.out.println(new InClass().num);//将静态内部类实例化获得其非静态变量
2.静态内部类只能外部类的静态变量,无法获得非静态变量。
// System.out.println(StaticInclass.num1);静态方法同样无法获取外部类的非静态变量
// System.out.println(StaticInclass.num1);非静态方法无法获取外部类的非静态变量
2>.非静态内部类
非静态内部类注意一点:内部类获取外部类的变量:“外部类.this.变量名”。
代码:
package com.cjm.inclass; public class InClassText {
int num; public InClassText(int num) {
this.num = num;
} public class InClass {
int num; public InClass(int num) {
this.num = num;
System.out.println(InClassText.this.num);//获取了外部类的变量
} @Override
public String toString() {
return "内部类的num:" + this.num;
}
} @Override
public String toString() {
return "外部类的num:" + this.num ;
} public static void main(String[] args) {
InClassText inclasstext = new InClassText(0);//外部类的初始化
InClassText.InClass inclass = inclasstext.new InClass(10);//内部类的初始化
}
}
3>.局部内部类
创建在方法或者代码块中的内部类为局部内部类
4>.匿名内部类
个人认为匿名内部类存在着打乱Java代码整体规则,使得代码混乱的问题,强烈推荐使用lambda表达式来代替匿名内部类。
匿名内部类的语法有些特别,创建匿名内部类时会立即创建一个该类的实例,这个类定义立即消失,匿名内部类不能重复使用。因此匿名内部类适合创建那种只需要一次使用的类。
匿名内部类的格式如下:
new 父类构造器|实现接口(){
//匿名内部类的类体部分
}
匿名内部类必须继承一个父类,或实现一个接口,但最多只能继承一个父类,实现一个接口。
- 匿名内部类不能是抽象类,因为系统在创建匿名内部类时,会立即创建匿名内部类的对象。
- 匿名内部类不能定义构造器,因为匿名内部类没有类名,也就无法定义构造器,但是匿名内部类可以定义实例初始化块,通过初始化块来完成初始化操作。
package com.cjm.inclass; /**
* 匿名内部类
*
* @author 小明
*
*/
public class NoInClass { public Myinterface getOne() {//创建一个匿名内部类,匿名内部类必须继承一个父类,或实现一个接口,但最多只能继承一个父类,实现一个接口。
return new A() {
/*
* 在A类中进行添加的方法无法实现
*/
private int num = 10;
public void f3() {
System.out.println("f3()");
}
};
} public static void main(String[] args) {
NoInClass a=new NoInClass();
Myinterface ain=a.getOne();
ain.f();
Myinterface.f2();
}
}
interface Myinterface {
void f();
int num=0;
static void f2(){//匿名内部类无法使用静态方法,对象无法调用静态方法
System.out.println("hello");
}
}
class A implements Myinterface{
@Override
public void f() {
// TODO Auto-generated method stub
System.out.println("重写了f方法!");
} }