Java内部类
Java内部类分类
1. 成员内部类
相当于成员,和成员定义位置相同
public class Test {
public static void main(String[] args) {
//第一种方式:
Outter outter = new Outter();
Outter.Inner inner = outter.new Inner();
//必须通过Outter对象来创建
inner.showInfo();
//第二种方式:
Outter.Inner inner1 = outter.getInnerInstance();
inner1.showInfo();
}
}class Outter {
private int count = 10;
private int outterCount = 10;
private Inner inner = null;
public Outter() {}public Inner getInnerInstance() {
if (inner == null)
inner = new Inner();
System.out.println(inner.count);
return inner;
}class Inner {
private int count = 20;
public Inner() {}public void showInfo() {
System.out.println(count);
System.out.println(outterCount);
}
}
}
- 成员内部类由于包含外部类实例,故可访问外部类成员和方法即使是私有
- 当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要
Outter.this.xxx
- 外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问,可以访问私有变量和方法
- 成员内部类是外部类成员,故要创建成员内部类的对象,前提是必须存在一个外部类的对象。有如下两种创建方式:
//第一种方式:
Outter outter = new Outter();
Outter.Inner inner = outter.new Inner();
//必须通过Outter对象来创建
//第二种方式:
Outter.Inner inner1 = outter.getInnerInstance();
- 外部类只能被public和包访问两种权限修饰,但成员内部类拥有成员的修饰符特性,有4种 public,private,protected,默认
相当于局部变量,定义位置和局部变量位置相同,定义在作用域(方法或者代码块内)
class Outter {
public void method() {
final int count = 20;
class LocalInner {
public void method() {
System.out.println(count);
}
}
new LocalInner().method();
}}
- 和局部变量一样不能使用任何的访问修饰符。
- 会生成两个.class文件,一个是Outter.class ,另一个是Outter$LocalInner.class。
- 局部内部类只能访问方法中声明的final类型的变量。
- 我们是无法在外部去创建局部内部类的实例对象的,因为局部内部类是定义在方法中的,而方法是需要所在类的对象去调用。
3. 静态内部类
class Outter {
private static int a = 20;
public Outter() {}static class Inner {
public Inner() {
System.out.println(a);
}
}
}
- 静态内部类就是一个普通类,只是位置在另一个类的内部,并不持有外围类的引用
- 只能访问外部类的静态成员变量或者静态方法
- 会生成两个.class文件,一个是外部的类Outter.class , 另一个是 Outter$Inner.class
- 本质:匿名内部类会隐式的继承一个类或者实现一个接口,或者说,匿名内部类是一个继承了该类或者实现了该接口的子类匿名对象。
- 格式:
new 类名/接口/抽象类(){}例如:new Interface(){}
- 实例:
public class Test {
public static void main(String[] args) {
// 匿名内部类1 如果Air是类,则根据参数使用对应构造方法
Air air = new Air("rocket") {
@Override
void fly() {
System.out.println("fly");
}
};
air.fly();
// 匿名内部类2 该匿名类使用类的默认构造方法也就是空构造方法
Fly fly = new Fly() {
@Override
public void fly() {
System.out.println("fly");
}
};
fly.fly();
}
}abstract class Air {
public Air(String s) {
System.out.println(s);
}abstract void fly();
}interface Fly {
void fly();
}
输出
rocket
fly
fly
编译代码后匿名内部类1会生成Test$1.class, 反编译该字节码得到
static class Test$1 extends Air
{void fly()
{
System.out.println("fly");
}Test$1(String s)
{
super(s);
}
}
【Java内部类】自动生成了带参数的构造方法,并调用了父类的同参数的构造方法
- 特性
- 匿名内部类由于没有名字,故无法定义构造方法,但编译后会生成构造方法根据传入的参数
- 匿名内部类编译后生成的.class文件的命名方式是”外部类名称$编号.class”,编号为1,2,3…n,编号为x的文件对应的就是第x个匿名类
- 匿名内部类
- 可以弥补没有多继承的缺陷
public class Test {
public static void main(String[] args) {
Swan swan = new Swan();
swan.fly();
swan.swim();
}
}abstract class FlyingAnimal {
abstract void fly();
}abstract class SwimmingAnimal {
abstract void swim();
}class Swan extends FlyingAnimal {@Override
void fly() {
System.out.println("Swan.fly()");
}void swim() {
this.getSwimming().swim();
}SwimmingAnimal getSwimming() {
return new SwimmingAnimal() {@Override
void swim() {
System.out.println("Swan.swim()");
}};
}
}
Swan继承了FlyingAnimal,无法再继承SwimmingAnimal,此时可通过如上匿名内部类方式解决问题
但是,这种设计方式,有明显的缺陷。因为,Swan类本身只继承了FlyingAnimal类,所以它和FlyingAnimal是is - a的关系,它和SwimmingAnimal并没有继承关系,所以并不是is-a的关系,其实更像一种has - a的关系。所以,这并不符合常理。
- 解决命令冲突(例如抽象类的方法和接口中的方法同名)
- 方便将存在一定逻辑关系的类组织在一起,又可以对外界隐藏
解决办法:
将该变量可以生命为数组类型
eg:
final int[] counter = new int[1];
这样可以使counter中的内容发生改变
参考 http://www.cnblogs.com/dolphin0520/p/3811445.html
推荐阅读
- JAVA(抽象类与接口的区别&重载与重写&内存泄漏)
- 期刊|期刊 | 国内核心期刊之(北大核心)
- 你有婚内虐待行为吗()
- 标签、语法规范、内联框架、超链接、CSS的编写位置、CSS语法、开发工具、块和内联、常用选择器、后代元素选择器、伪类、伪元素。
- ?【段子图】内裤为啥湿呢(想想好邪恶啊...)
- 事件代理
- 放下心中的偶像包袱吧
- Java|Java OpenCV图像处理之SIFT角点检测详解
- java中如何实现重建二叉树
- angular2内置管道