非静态内部类作用:

  1. 最基本的作用:名字隐藏和组织代码

    • 用例:内部类不访问外部类的元素时可以直接new。(bad style!)
    • 用例:通过外部类的非静态方法返回内部类的引用,这样隐含了内部类对象和其对应的外部类对象的关系(需要先创建外部类对象,然后通过外部类对象创建内部类对象)。
    • 注意:内部类访问内部类的元素时,直接new内部类编译会报错

      No enclosing(外围的) instance of type xxx is accessible
  2. 链接到外部类:访问其外围类所有成员的访问权
    • 实现机制:普通内部类(非静态)对象和外部类对象之间存在联系。创建内部类对象时,该对象保存指向其外部类对象的引用。
    • 用例:"迭代器"设计模式

非静态内部类使用:

  1. 外部类类名 + .this

    • 用法:返回内部类对象引用的外部类对象
  2. 外部类对象 + .new

    • 用法:返回内部类的引用,告知其它对象或静态方法中使用
    // Creating an inner class directory using the .new syntax
    
    public class DotNew {
    public class Inner {}
    public static void main(String[] args) {
    DotNew dotNew = new DotNew();
    DotNew.Inner inner = dotNew.new Inner();
    }
    }
  3. 权限修饰符

    1. 私有内部类,隐藏内部类信息。接口的实现不可见也不可用,仅能得到指向基类或者接口的引用。
    2. 保护内部类,自己、子类、同一个包中的类可以访问
    // 接口所有成员自动是public的
    interface Destination {
    String readLabel();
    } interface Contents {
    int value();
    } class Parcel4 {
    // 私有内部类,隐藏内部类信息。接口的实现不可见也不可用,仅能得到指向基类或者接口的引用。
    private class PContents implements Contents {
    private int i = 11;
    public int value() { return i; }
    }
    // 保护内部类,自己、子类、同一个包中的类可以访问
    protected class PDestination implements Destination {
    private String label;
    private PDestination(String whereTo) {
    this.label = whereTo;
    }
    public String readLabel() { return this.label; }
    }
    public Destination destination(String s) {
    PDestination pDestination = new PDestination(s);
    // Exercise 8: 外部类是否可以访问内部类的private元素
    // 答:外部类访问内部类的private对象就像访问自己的private对象一样
    // 理由:编译器对内部类实现时,生成两个类,对于private变量的访问,编译器会生成一个accessor函数.
    System.out.println(pDestination.label);
    return pDestination;
    }
    public Contents contents() {
    return new PContents();
    }
    } public class TestParcel {
    public static void main(String[] args) {
    Parcel4 p = new Parcel4();
    Contents c = p.contents();
    Destination d = p.destination("Tasmania");
    // The type Parcel4.PContents is not visible
    // 不能向下转型成private内部类,因为名字不可访问
    // ! Parcel4.PContents pc = p.new PContents();
    }
    }
    • 注意:嵌套类(内部静态类)不需要对外部对象的引用
    // Exercise 8: 外部类是否可以访问内部类的private元素
    // 答:可以,外部类访问内部类的private对象就像访问自己的private对象一样
    // 实现机制:编译器对内部类实现时,生成两个类,对于private变量的访问,编译器会生成一个 accessor函数.

方法和作用域内的内部类

  1. 目的

    • 实现了某种类型的接口,可以创建并返回对其的引用
    • 创建一个类以解决问题,但不希望这个类是公共可用的
  2. 局部内部类

    • 在方法内部定义类
    • 局部内部类可以嵌入某个作用域内(如if作用域),则在作用域外不可访问
  3. 匿名内部类

    • 创建继承/实现自公共类/接口的匿名类的对象
    // Returning an instance of an anonymous inner class.
    
    // 普通类被当作“公共”接口来使用
    class Wrapping {
    private int i;
    public Wrapping(int x) { i = x; }
    public int value() { return i; }
    } public class Parcel7 {
    public Contents contents() {
    // 创建一个实现/继承自Contents的匿名类的对象,且使用默认的构造器
    return new Contents() {
    private int i = 11;
    public int value() { return i; }
    };
    } public Wrapping wrapping(int x) {
    // Base constructor call:
    return new Wrapping(x) { // Pass constructor argument
    public int value() {
    return super.value() * 47;
    }
    };
    } public static void main(String[] args) {
    Parcel7 parcel7 = new Parcel7();
    Wrapping wrapping = parcel7.wrapping(4);
    System.out.println(wrapping.value());
    }
    }
    public class Parcel9 {
    // Argument must be final to use inside
    // anonymous inner class:
    public Destination destination(final String dest) {
    return new Destination(){
    private String label = dest;
    @Override
    public String readLabel() {
    return label;
    }
    };
    }
    public static void main(String[] args) {
    Parcel9 p = new Parcel9();
    Destination d = p.destination("Tasmania");
    System.out.println(d.readLabel());
    }
    }
    • 匿名内部类、局部内部类:若要使用一个在其外部定义的对象,那么编译器会要求其参数引用是final的

    Java编译器实现的只是capture-by-value,并没有实现capture-by-reference

        interface AnnoInner(){addXYZ();}
    public class Outer {
    public AnnoInner getAnnoInner(final int x) {
    final int y = 100;
    return new AnnoInner(){
    // 编译器相当于拷贝外部自由变量的副本到匿名类
    // int copyX=x;
    // int copyY=y;
    int z = 100;
    public int addXYZ() {
    return x + y + z;
    }
    }
    }
    }
    • 利用实例初始化,以达到为匿名内部类创建一个构造器的效果
    • 注意:匿名内部类既可以扩展类,也可以实现接口,但不能同时扩展类和实现接口,且如果是实现接口,也只能实现一个接口

嵌套类

不需要内部类对象与其外部类对象之间有联系。

  1. 意义

    • 要创建嵌套类的对象,并不需要其外围类的对象
    • 不能从嵌套类的对象中访问非静态的外围类对象
    • 嵌套类可包含static数据和字段,但普通内部类不行

Java编程思想读书笔记 第十章 内部类的更多相关文章

  1. 《Java编程思想》笔记 第十章 内部类

    1.创建内部类对象 创建内部类对象(相当于new内部类对象) 外围类对象.new 内部类( ). 创建内部类对象一种方法是 外围类有一个方法返回内部类对象. 没有外围类对象就不能创建内部类对象,因为内 ...

  2. JAVA编程思想读书笔记(五)--多线程

    接上篇JAVA编程思想读书笔记(四)--对象的克隆 No1: daemon Thread(守护线程) 参考http://blog.csdn.net/pony_maggie/article/detail ...

  3. JAVA编程思想读书笔记(四)--对象的克隆

    接上篇JAVA编程思想读书笔记(三)--RTTI No1: 类的克隆 public class MyObject implements Cloneable { int i; public MyObje ...

  4. JAVA编程思想读书笔记(三)--RTTI

    接上篇JAVA编程思想读书笔记(二) 第十一章 运行期类型判定 No1: 对于作为程序一部分的每个类,它们都有一个Class对象.换言之,每次写一个新类时,同时也会创建一个Class对象(更恰当的说, ...

  5. JAVA编程思想读书笔记(二)--容器

    接上篇JAVA编程思想读书笔记(一) 第八章.对象的容纳 No1: java提供了四种类型的集合类:Vector(矢量).BitSet(位集).Stack(堆栈).Hashtable(散列表) No2 ...

  6. Java编程思想读书笔记之内部类

    现在是够懒得了,放假的时候就想把这篇笔记写出来,一直拖到现在,最近在读<Java编程思想>,我想会做不止这一篇笔记,因为之前面试的时候总会问道一些内部类的问题,那这本书的笔记就从内部类开始 ...

  7. Java编程思想读书笔记(一)【对象导论】

    2018年1月7日15:45:58 前言 作为学习Java语言的经典之作<Java编程思想>,常常被人提起.虽然这本书出版十年有余,但是内容还是很给力的.很多人说这本书不是很适合初学者,我 ...

  8. Java编程思想读书笔记

    声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...

  9. <Java编程思想>读书笔记(1)-对象导论、一切都是对象

    1.面向对象编程:OOP (Object-oriented Programming) 2.Alan Kay 总结的面向对象语言5个基本特性: 1) 万物皆为对象 2) 程序是对象的集合,他们通过发送消 ...

随机推荐

  1. 理解ext文件系统

    理解ext文件系统 @(0001学习博客) 注意:本文参考骏马金龙的博客,详情请移步浏览 一.一些常见的文件系统 Linux的文件系统: ext2(无日志功能), ext3, ext4, xfs, r ...

  2. netty实现客户端服务端心跳重连

    前言: 公司的加密机调度系统一直使用的是http请求调度的方式去调度,但是会出现网络故障导致某个客户端或者服务端断线的情况,导致很多请求信息以及回执信息丢失的情况,接着我们抛弃了http的方式,改为T ...

  3. Android_(控件)动态添加或删除Spinner下拉菜单

    使用ArrayList动态数组的依赖性实现动态增减Spinner下拉菜单选项功能. 设置一个EditText输入框,当用户输入了文字并单击[添加]按钮的同时,就会将输入的值添加Spinner至下拉菜单 ...

  4. dependencies和devDependencies的区别?

    当我们项目需要下载一个模块的时候,我们安装npm包(在项目目录下面npm install module_name)的时候,很多时候我们会在后面加上–save-dev 或 –save.这两个参数代表什么 ...

  5. html5 代码画兰博基尼跑车,6不6你说的算!

    源代码下方 由于本人喜爱html5,无聊所画: 画图需要掌握; 1.画布,画笔,画圆,给画笔添加颜色.(注:掌握这几点,你就可以称霸画图界了.) 虽然没有画画天赋,但代码写的也是溜溜滴!(注:此图没有 ...

  6. IDEA的Python工程在inspections时提示:The JDK is not configured properly for this project. Inspection cannot proceed.

    原因:IDEA将工程误认为了Java工程. 解决方法:如果确认这是个Python工程,修改*.iml文件,将下面的这一行删除 <orderEntry type="inheritedJd ...

  7. Linux常用命令及操作

    shutdown -r now现在重启 shutdown -h now现在关机 reboot重启 startx进入图形界面 chmod +x 777 文件名 授权和可执行 777为二进制111-111 ...

  8. Maven项目解决Remove '@override' annotation终极方案

    特别提示:本人博客部分有参考网络其他博客,但均是本人亲手编写过并验证通过.如发现博客有错误,请及时提出以免误导其他人,谢谢!欢迎转载,但记得标明文章出处:http://www.cnblogs.com/ ...

  9. 两个 Activity 之间跳转时必然会执行的是哪几个方法?

    一般情况下比如说有两个 activity,分别叫 A,B,当在 A 里面激活 B 组件的时候, A 会调用 onPause()方法,然后 B 调用 onCreate() ,onStart(), onR ...

  10. 细说Linux下的虚拟主机那些事儿

    细说Linux下的虚拟主机那些事儿 我们知道Linux操作系统是目前在服务器上应用广泛的操作系统.在Linux操作系统下的虚拟主机是不是就是我们常说的Linux虚拟主机呢?其实从专业方面说并不是,它是 ...