什么

定义在一个类内部的类,称为内部类(累不累),如下:

public class A {
    private int c = 1;

    public class C {
        public void test() {
            System.out.println("c:" + c);
        }
    }
}

C称为A的内部类,简称内部类

A称为C的外部类,简称外部类

而且内部类能访问外部类的成员(静态成员、实例成员),当然有一些限制,限制如下

4种声明方式

按照内部类的声明方式,分为4种内部类:

  1. 静态内部类

    像类的静态成员一样声明的类,就称呼为“静态内部类”

    public class A {
        private static String b = "b";
            private int c = 1;
    
            // B是A的静态内部类
        public static class B {
            public void test() {
                System.out.println(b);
            }
        }
    }  

    静态内部类,只能访问外部类的静态成员(方法和变量),并且可以像类的成员一样使用修饰符(public/protected/private);

    创建静态内部类对象的方式:A.B b = new A.B();

  2. 成员内部类

    新类的实例成员(未加static修饰)声明的类,称为“成员内部类”

    public class A {
        private static String b = "b";
            private int c = 1;
    
            // C是A的成员内部类
        public class C {
            public void test() {
                System.out.println(c);
                            System.out.println(b);
            }
        }
    }

    成员内部类,访问外部类的一切(静态,还是实例),就像成员方法一样,并且可以像类的成员一样使用修饰符(public/protected/private)

    创建成员内部类对象的方式:

    A a = new A();
    A.C c = a.new C();
  3. 方法内部类

    在一个代码块声明的类称为方法内部类,代码块包括(方法内、静态代码块内、实例代码块内)

    public class A {
        private static String b;
        private int c;
    
            // 成员方法
        public void test() {
            final int d = 1;
            // 方法内部类
            class D {
                public void test() {
                    // 访问静态变量
                    System.out.println(b);
                    // 访问实例变量
                    System.out.println(c);
                    // 访问方法final类型的局部变量
                    System.out.println(d);
                }
            }
        }
    }

    方法内部类,和它所在的方法(代码块),具有相同的访问能力,如果上面代码是在static方法中声明的,那么内部类D不能访问c变量。

    jdk1.8 方法内部类,能够访问非final类型的局部变量,本质相当有在内部类D内保存了副本

  4. 匿名内部类

    匿名内部类也就是没有名字的内部类

    正因为没有名字,所以匿名内部类只能使用一次,它通常用来简化代码编写

    但使用匿名内部类还有个前提条件:必须继承一个父类或实现一个接口

内部类的本质

内部类的语法颇为奇怪,我们来看看如下代码,编译后的字节码文件!

public class A {
    private static String b = "b";
    private int c = 1;
        // 静态内部类
    public static class B {
        public void b() {
            System.out.println(b);
        }
    }
        // 成员内部类
    class C {
        public void c() {
            System.out.println(c);
        }
    }
}

  1. A.java文件被编译成了多个class文件
  2. A类对应A.class
  3. B类对应A$B.class
  4. C类对应A$C.class

内部类会被编译成单独的class文件,那意味JVM解释执行class文件时类“B”和类A是独立的,由此可以见内部类也是一种语法糖!

对于JVM来说,类A的private b和c 成员,怎么能分别被类B和类C访问到的了!

用javap命令反编译类A.class来看看:

秘密就来自,编译器为外部类生的两个静态访问方法,Stinrg access$000()返回b变量的值,int access$100(A a)返回a对象的c成员变量值;

而在静态内部类B中,编译器将访问静态变量b的地方替换为如上方法:

// 静态内部类
public static class A$B {
    public void b() {
        System.out.println(A.access$000());
    }
}

在成员内部类C中,原理也是如此,不过增加了更多的东西,反编译A$C.class:

  1. 新增了成员字段final A $this;
  2. 构造方法添加形参 `A$C(A obj);
  3. 访问外部类成员变量的地方会被替换成:System.out.println(A.access$100($this));

你一定会好奇成员构造方法中的外部类对象的参数从哪里传入的!看看我们是怎么声明内部类的对象的

A a = new A();
A.C c = a.new C();

将会被编译器替换成:

A a = new A();
A$C c = new A$C(a);

内部类的使用时机

两个类之间紧密联系时,可以使用内部类:

  1. 当一个类需要访问另外一个类的许多属性时,内部类可以简化访问代码
  2. 实现更好封装性,比如:B 类仅仅被A类访问时,可以将B类作为A的私有内部类
  3. 使代码更简洁,匿名内部类

Java 内部类详解的更多相关文章

  1. Java内部类详解

    Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...

  2. [转] Java内部类详解

    作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本博客中未标明转载的文章归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置 ...

  3. 【转】Java内部类详解

    一.内部类基础 在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类.广泛意义上的内部类一般来说包括这四种:成员内部类.局部内部类.匿名内部类和静态内部类.下面就先来了解一 ...

  4. Java内部类详解(一)

    (转自:http://blog.csdn.net/wangpeng047/article/details/12344593) 很多人对于Java内部类(Inner Class)都十分陌生,甚至听都没听 ...

  5. Java内部类详解 2

    Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...

  6. 【Java_基础】Java内部类详解

    1.四种内部类 java中的四种内部类:成员内部类.静态内部类.局部内部类和匿名内部类.其中匿名内部类用到的最多. 1.1.成员内部类 若一个类定义在另一个类的内部作为实例成员,我们把这个作为实例成员 ...

  7. Java内部类详解(转)

    说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就来一探究竟.下面是本 ...

  8. (转)java内部类详解

    本文转自http://www.cnblogs.com/dolphin0520/p/3811445.html,谢谢作者 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能 ...

  9. Java——内部类详解

    说起内部类,大家肯定感觉熟悉又陌生,因为一定在很多框架源码中有看到别人使用过,但又感觉自己使用的比较少,今天我就带你具体来看看内部类. 内部类基础 所谓内部类就是在类的内部继续定义其他内部结构类. 在 ...

随机推荐

  1. UVa 750 - 8 Queens Chess Problem

    题目大意:八皇后问题,在一个8*8的棋盘上,放置8个皇后,使得任意两个皇后不在同一行上.不在同一列上.不在同一条对角线上,不过这道题预先给定了一个位置放置一个皇后,让你输出所有可能的答案. 经典的回溯 ...

  2. ubuntu系统内核替换

    此处将内核由高版本替换成低版本.替换前的系统为ubuntu 12.04 kernel 3.8.0. 替换后的内核版本为2.6.35. 首先下载需要替换的内核文件,下载链接:https://www.ke ...

  3. 十二生肖&天干地支

    看相:http://baike.baidu.com/view/833122.htm?fr=aladdin 八字:http://baike.baidu.com/view/17127.htm?fr=ala ...

  4. 因子分析&主成分分析

    因子分析和主成分分析的异同点: 1.主成分分析仅仅是一种数据变换而不假设数据矩阵有什么样的结构形式 因子分析假定数据有特定的模型,而且齐总的因子满足特定的条件 2.因子分析和主成分分析都是从相关矩阵出 ...

  5. Zepto swipe 无效(坑)

    Zepto 滑动插件 bug Zepto 的 'swipe', 'swipeLeft', 'swipeRight', 'swipeUp', 'swipeDown' 触摸事件在安卓4.4系统中除chro ...

  6. Angular - - $interval 和 $timeout

    $interval window.setInterval的Angular包装形式.Fn是每次延迟时间后被执行的函数. 间隔函数的返回值是一个承诺.这个承诺将在每个间隔刻度被通知,并且到达规定迭代次数后 ...

  7. iOS 之 assign、retain、copy、nonatomic

    1. assign 1.1. 普通赋值 一般用于基本类型 1.2. 常见委托设计模式 防止循环引用 2. retain 保留计数,获取了对象的所有权.引用计数在原有基础上加1. 3. copy 同re ...

  8. --@angularJS--指令之单个点击展开demo

    1.expander.html: <!DOCTYPE HTML><html ng-app="app"><head>    <title&g ...

  9. Iframe 自适应高度并实时监控高度变化的js代码

    不得不用到iframe,且被强烈要求不能让它出现滚动条!嵌入的页面肯定是高度不一的,页面中也不能出现大片空白,所以也不能写死高度!真是麻鬼烦啊!google N次 + 百度M次 + 试验了1605次之 ...

  10. HDU-1754-I Hate It(线段树,简单,不过好像有点问题)

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=1754 题目不难,不过开始我犯了一个低级错误,输入n,m,m代表操作的数目,我没有写了,写代码的时候,就 ...