Think in java 读书笔记

pikzas

2019.05.05

第十章 内部类

知识点

1.什么是内部类

可以将一个类定义在另一个类的内部

class OuterClass{
class InnerClass{ }
}

2.内部类的主要特征是什么

内部类的主要特征是内部类可以随意访问外部类的属性和方法,不论其访问修饰符是什么

public interface Selector {
boolean end();
Object current();
void next();
} public class Seq {
private Object[] items;
private int next = 0;
public Seq(int size){
items = new Object[size];
}
public void add(Object item){
if(next < items.length){
items[next++] = item;
}
} private class SeqSelector implements Selector{
private int i = 0; @Override
public boolean end() {
return i == items.length;
} @Override
public Object current() {
return items[i];
} @Override
public void next() {
if(i<items.length){
i++;
}
}
} public Selector getSelector(){
return new SeqSelector();
} public static void main(String[] args) {
Seq sq = new Seq(10);
for (int i =0 ; i < sq.items.length ; i++){
sq.add(Integer.toString(i));
}
Selector selector = sq.getSelector(); //此时,外部类对象的引用会被传递给内部类对象
while (!selector.end()){
System.out.println(selector.current());
selector.next();
}
} }

3.适用于内部类的一些特殊语法

.new 由外部类对象新创建内部类对象

class Outer{
class Inner{}
} class Demo{
public static void main(String[] args){
Outer outer = new Outer();
Outer.Inner inner = outer.new Inner();
}
}

.this 在内部类如果想要获取外部类对象的引用

class DotThis{
void f() {System.out.println("DotThis.f()");}
class Inner{
public DotThis getOuterObj(){
return DotThis.this;
}
} public Inner getInner(){
return new Inner();
} public static void main(String[] args){
DotThis dotThis = new DotThis();
DotThis.Inner inner = dotThis.getInner();
System.out.println(inner.getOuterObj().equals(dotThis)); // 输出为true
inner.getOuterObj().f(); // inner.getOuterObj() 等同于 dotThis }
}

.new 语法也可已看出,必须要先有外部类实例,再由这个实例.new 出内部类。(静态内部类除外)

4.内部类与接口的配合使用

内部类实现接口,并将内部类访问限定为private或者protected,从而将实现隐藏在该内部类中。

public interface Dest {
int value();
} public class Demo {
private class DestImp implements Dest{
@Override
public int value() {
return 0;
}
} public Dest getDest(){
return new DestImp();
}
} public class Test {
public static void main(String[] args) {
Demo demo = new Demo();
Dest dest = demo.getDest();
dest.value();
}
}

5.局部内部类

以上看到的内部类都是直接定义在外部类的作用域内,如果让一个内部类定义在外部类的某个方法内部时,这就称作局部内部类。

并且当作用域超出了这个方法的时候,外部是访问不到该类的。

public interface Fruit {
String desc();
} public class Test {
public Fruit getFruit(){
class Apple implements Fruit{
@Override
public String desc() {
return "i am apple";
}
}
return new Apple();
}
// getFruit方法之外,是访问不到Apple类的
// private Fruit fruit = new Apple();
}

6.匿名内部类

上面的局部内部类将Apple的定义和new 创建新对象的语法放在一起就是匿名内部类。如果返回的是具体的类,看起来也就没什么特殊了。

public interface Fruit {
String desc();
} public class Test2 {
public Fruit getFruit(){
return new Fruit(){
@Override
public String desc() {
return "i am apple";
}
};
}
public static void main(String[] args) {
Test2 test2 = new Test2();
Fruit fruit = test2.getFruit();
System.out.println(fruit.desc());
}
}

6.1.匿名内部类可以是具体类,甚至可以对其方法进行override

public class Wrapping {
private int i;
public Wrapping(int x){
this.i = x;
}
public int value(){
return i;
}
} public class Outer {
public Wrapping getItem(int x){
return new Wrapping(x){
@Override
public int value(){
return super.value() * 47;
}
};
} public Wrapping getItem2(int x){
return new Wrapping(x);
} public static void main(String[] args) {
Outer outer = new Outer();
Wrapping wrapping = outer.getItem(10);
Wrapping wrapping2 = outer.getItem2(10);
System.out.println(wrapping.value());
System.out.println(wrapping2.value());
}
} ------输出结果------
470
10

6.2.对匿名内部类的字段进行初始化

  • 使用外部类传入的数据进行初始化,此时该数据必须为final的
public interface SampleInterface {
int value();
} public class InitDemo {
public SampleInterface getSample(final int x){ //此处final必须为final的 可以理解为外部对象指向的数据不能变动,否则内部类对象会很困惑。
return new SampleInterface() {
private int i = x;
@Override
public int value() {
return i;
}
};
}
}
  • 通过初始化代码块完成类似于构造器的功能
public abstract class Base {
public Base(int i){
System.out.println("base constructor, i = " + i);
}
public abstract void f();
} public class AnonymousConstructor {
public static Base getBase(int i){ //此处的变量不用是final的,因为内部类并没用到i
return new Base(i){
{
System.out.println("inside instance"); // 通过静态代码块可以实现初始化
}
public void f(){
System.out.println("in inner f()");
}
};
} public static void main(String[] args) {
Base base = getBase(47);
base.f();
}
}

从上面对于传入内部类的参数是否要加上final的修饰规定是,如果内部类用到了外部传入的那个变量,则需要为final的。

内部类可以拿来扩展类,也可以拿来实现接口,但是不能两个都做到,也只能一次实现一个接口。

7.工厂方法的内部类实现方式

public interface Game {
boolean move();
} public interface GameFactory {
Game getGame();
} public class Chess implements Game {
private Chess(){};
private int moves = 0;
private static final int MOVES = 4;
@Override
public boolean move() {
System.out.println("Chess moves" + moves);
return ++moves != MOVES;
} public static GameFactory factory = new GameFactory(){
public Game getGame(){
return new Chess();
}
}; } public class Checkers implements Game {
private Checkers(){}
private int moves = 0;
private static final int MOVES = 3; @Override
public boolean move() {
System.out.println("Checker moves" + moves);
return ++moves != MOVES;
} public static GameFactory factory = new GameFactory() {
@Override
public Game getGame() {
return new Checkers();
}
};
} public class Test {
public static void playGame(GameFactory factory) {
Game game = factory.getGame();
while(game.move());
}
public static void main(String[] args) {
playGame(Chess.factory);
playGame(Checkers.factory);
}
}
-------输出结果---- Chess moves0
Chess moves1
Chess moves2
Chess moves3
Checker moves0
Checker moves1
Checker moves2

8.嵌套类(也成为静态内部类)

前面讲到的内部类的创建都依赖于外部类的对象,如果内部类定义的时候是static的,那么外部对象就不需要了。

同时因为内部类是static的,那么如果想要使用外部类中的属性或者方法,那么那些属性或者方法必须也是static的。


public class Outer {
private int x = 123;
private static int y = 999;
public static class OneImpl implements OneInterface{
public void f(){
// System.out.println("inner class" + x); 此处因为x不是static的,会提示编译错误。
System.out.println("inner class " + y);
}
} public static void main(String[] args) {
OneInterface one = new OneImpl();
one.f();
}
}

8.1.嵌套类的应用

由于接口中的属性和方法默认都是public static的,所以其中可以来嵌套一些需要子类实现都通用的方法。

public interface DemoInterface {
void fun();
class DemoImpl implements DemoInterface{
public void fun(){
System.out.println("可以实现我自己的外部接口");
}
}
} public class DemoTest implements DemoInterface {
@Override
public void fun() {
System.out.println("DemoTest");
} public static void main(String[] args) {
DemoInterface demo = new DemoTest();
DemoInterface demo2 = new DemoImpl();
demo.fun();
demo2.fun();
}
} -----输出结果-----
DemoTest
可以实现我自己的外部接口

嵌套类不同于普通的内部类的另一点在于,编译生成的.class文件名称都是形如Outer$Inner.class这个格式。

而且嵌套类内部还能有嵌套类,以及static声明的属性及方法。

9.内部类无论内部有多少层,内一层的类都可以无条件的访问外部类

10.内部类存在的意义

内部类存在的主要目的是为了解决java没有多继承导致的一些麻烦

  • java可以很好的解决多接口的实现问题,如下例子
public interface A {
} public interface B {
} public class X implements A,B {
} public class Y implements A {
B getB(){
return new B() {};
}
} public class TestOne {
public static void funA(A a) { } public static void funB(B b) { } public static void main(String[] args) {
X x = new X();
Y y = new Y();
funA(x);
funA(y);
funB(x);
funB(y.getB());
}
}
  • 但是对于要求子类同时实现抽象类和父类问题时,只有内部类可以派上用场
public class M {
} public abstract class N {
} public class Z extends M {
N getN(){
return new N() {};
}
} public class TestTwo {
public static void funM(M m) { }
public static void funN(N n) { }
public static void main(String[] args) {
Z z = new Z();
funM(z);
funN(z.getN());
}
}

11.内部类的继承

内部类与外部类紧紧联系在一起,那么如果想要继承一个外部类的内部类,就需要使用到特殊的语法 OutClass.super();

public class DemoOuter {
class Inner{}
} public class ExtInner extends DemoOuter.Inner {
public ExtInner(DemoOuter demoOuter){
demoOuter.super(); //必须添加如此这般的构造器并显示指定父类对象引用
}
}

12.内部类初始化顺序

内部类的构造器会先调用,然后才会调用外部类构造器。

内部类不存在覆盖问题,Egg中的Yolk和BigEgg中的Yolk完全是两个不同的类。

public class Egg {
class Yolk{
public Yolk(){
System.out.println("Egg York"); // 1 3
}
public void f(){
System.out.println("Egg f()");
}
}
private Yolk york = new Yolk();
public Egg(){
System.out.println("Egg"); // 2
}
public void insert(Yolk y){
this.york = y;
}
public void g(){
york.f();
}
} public class BigEgg extends Egg {
class Yolk extends Egg.Yolk{
public Yolk(){
System.out.println("BigEgg Yolk"); // 4
}
public void f(){
System.out.println("BigEgg f()"); // 5
}
} public BigEgg(){
insert(new Yolk());
} public static void main(String[] args) {
BigEgg egg = new BigEgg(); // 0
egg.g();
}
} -----输出结果-----
Egg York
Egg
Egg York
BigEgg Yolk
BigEgg f()

13.局部内部类和匿名内部类的区别

前面讲过,将显示的类声明写出来,在返回一个对象的方式叫做局部内部类,但是如果直接new 接口并返回,那么他就是匿名内部类。

如果某个场景下,需要返回一个接口的多种不同实现,那么只有局部内部类做得到,匿名内部类是做不到的。

14.内部类标识符

匿名内部类 --> LocalInnerClass$1.class

局部内部类 --> LocalInnerClass$Inner.class

最新文章

  1. SQL Server数据库级别触发器
  2. SpringMVC——自定义拦截器、异常处理以及父子容器配置
  3. git恢复误删文件及省去密码提交
  4. OS X 使用技巧——访问所有的键盘功能
  5. ios开发——实用技术总结Swift篇&amp;swift常用开发技术总结
  6. 1515 跳 - Wikioi
  7. UVA 11082 Matrix Decompressing 矩阵解压(最大流,经典)
  8. http://oncenote.com/2015/09/16/Security-2-HTTPS2/ (轉載)
  9. ie6的兼容总结
  10. 小学生之SpringMVC
  11. getgrent
  12. Google Play和基于Feature的过滤 —— Feature 参考手册
  13. diskpart修改盘符
  14. TFS 测试用例导入、导出工具
  15. OpenCascade Ruled Surface
  16. WordPress版微信小程序3.1.5版的新功能
  17. ipython安装( jupyter)
  18. tomcat和eclipse-wtp的一些配置
  19. java http post上传文件
  20. springboot中对yaml文件的解析

热门文章

  1. react 渲染
  2. app点击底部菜单切换标题
  3. AM335X的SD卡更新系统学习记录
  4. python解压压缩包的几种方式
  5. 线程同步器CountDownLatch
  6. WPF页面切换
  7. yii2模型
  8. keil密钥过期 新注册机 有效期至2032年
  9. argmax( )
  10. codeforces Codeforces Round #597 (Div. 2) Constanze&#39;s Machine 斐波拉契数列的应用