Java允许在一个类中定义另外一个类,这样的类被称为嵌套类,就像下面这样:

class OuterClass {
...
class NestedClass {
...
}
}

  嵌套类分为两种:静态的和非静态的。声明为static的嵌套类被称为静态嵌套类,非静态嵌套类则被称为内部类:

class OuterClass {
...
static class StaticNestedClass {
...
}
class InnerClass {
...
}
}

  嵌套类是其所在的外部类的成员。内部类可以访问外部类中的其他成员,即使这个成员被private修饰。静态嵌套类则没有访问外部类中其他成员的权限。作为外部类的一个成员,嵌套类可以被声明为private、public、protected或者包私有的。

  下面是几个为什么要使用嵌套的类原因:

  • 能够将仅在一个地方使用的类合理地组合。如果一个类可能只对于另外一个类有用,此时将前者组合到后者,可以使得程序包更加简洁。
  • 增强封装性。假如有两个类A和B,B类需要使用A类中的成员,而恰好该成员又是仅类内部可见(private)的,如果将B定义为A的嵌套类,则B可以使用A的任何成员,而且B也可以声明为外部不可见(private),将B隐藏起来。
  • 能够使代码可读性和维护性更强。嵌套的类代码相较于顶级类,更靠近它被使用的地方,方便查看。

一.静态嵌套类

  就像静态方法和静态变量一样,静态嵌套类是和外部类相关联的。和静态方法一样,静态嵌套类不能直接引用实例变量和实力方法,只能通过一个对象引用。实际上,可以将静态嵌套类看作是一个顶级类,只不过将其嵌套在其他类中方便打包。

  静态嵌套类可以用过外部类的名字去访问:

OuterClass.StaticNestedClass

  可以使用下面的语法为静态嵌套类创建对象:

OuterClass.StaticNestedClass nestedObject = new OuterClass.StaticNestedClass();

二.内部类

  就像实例方法和实例变量一样,内部类与外部类的实例相关联并且可以直接访问外部类的方法和成员。并且,因为内部类与外部类的实例相关联,因此它内部不能定义静态成员。

  要实例化内部类,必须创建外部类的对象,然后使用这个对象去创建内部类的对象:

OuterClass outerObject = new OuterClass();
OuterClass.InnerClass innerObject = outerObject.new InnerClass();

  实际上,还有两种特殊的内部类————局部类和匿名类。有关这两种类的内容将会在下文中介绍。强烈建议不要对内部类(包括局部类和匿名类)进行序列化(把对象转换为字节序列的过程称为对象的序列化,有关序列化的内容会在以后的文章中进行介绍)。

三.屏蔽现象

  如果一个类型(例如成员变量或参数)与外部作用域中的类型同名,那么内部作用域中的声明将会屏蔽外部作用域中的声明,这样就不能直接通过名称去访问外部作用域中同名的类型。如下例所示:

public class ShadowTest {
public int x = 0; class FirstLevel {
public int x = 1; void methodInFirstLevel(int x) {
System.out.println("x = " + x);
System.out.println("this.x = " + this.x);
System.out.println("ShadowTest.this.x = " + ShadowTest.this.x);
}
} public static void main(String... args) {
ShadowTest st = new ShadowTest();
ShadowTest.FirstLevel fl = st.new FirstLevel();
fl.methodInFirstLevel(23);
}
}

  这个例子的输出如下:

x = 23
this.x = 1
ShadowTest.this.x = 0

  这个例子中定义了三个名为x的变量,分别是ShadowTest类的成员变量,内部类FirstLevel的成员变量,以及方法methodInFirstLevel的参数。方法methodInFirstLevel的参数x屏蔽了内部类FirstLevel的成员变量x和ShadowTest类的成员变量x。因此,在表示内部类FirstLevel的成员变量x时,要像下面这样:

System.out.println("this.x = " + this.x);

  在表示ShadowTest类的成员变量x时,要像下面这样:

System.out.println("ShadowTest.this.x = " + ShadowTest.this.x);

四.局部类

  局部类是在块(由大括号包围的零条或多条语句)中定义的类。经常会在方法体中见到局部类。下面是一个定义在方法中的局部类:

public class OuterClass {
public void method() {
...
class LocalClass {
...
}
}
}

  局部类可以访问外部类的成员。此外,局部类还可以访问局部变量。然而,局部类只能访问由final修饰的局部变量。当一个局部类访问一个块中的局部变量或参数时,它就捕获了这个变量或者参数。从Java8开始,局部类不但可以访问由final修饰的局部变量和参数,还可以访实际上是final的局部变量和参数。实际上是final的意思是说这个变量或参数的值自从初始化之后就没有修改过。此外,局部类中的变量也会屏蔽定义在外部的同名变量或参数。

  局部类中基本不能定义静态成员。不过也有例外,可以在局部类中定义静态常变量(常变量是指类型为基本数据类型或者String,被声明为final,并且使用编译时常量表达式进行初始化的变量。编译时常量表达式通常是可以在编译时计算的字符串或算术表达式)。

  定义在静态方法中的局部类,只能引用外部类的静态成员。不能在块中定义接口,因为接口是天生静态的。也不能在局部类中定义静态初始化器或者接口。

五.匿名类

  匿名类让代码看上去更加简洁,它能让你同时声明和实例化一个类。它们类似于局部类,只不过匿名类没有名称。如果某个局部类只使用一次,可以将它定义为匿名类。

  局部类是类的声明,而匿名类则是表达式,这意味着匿名类是在一个表达式中定义的。匿名类表达式的语法就像是调用构造器的语法,只不过构造器后面跟的是类的定义。下面定义了一个Object类的子类,它重写了Object类的toString方法。由于这个类没有名字,因此只能在声明它的地方使用一次。虽然这个例子没什么意义,不过可以从中看到匿名类的语法,就像下面这样:

Object obj = new Object() {
@Override
public String toString() {
Sysyem.out.println("I'm a method from anonymous class.");
}
};

  匿名类可以继承自某个类,也可以实现某个接口。匿名类必须实现父类型中的抽象方法,至于父类型中的其他方法则视需求选择重写或不重写。如果匿名类中新增的方法不是来自父类,那么这个方法将无法访问。这是由于匿名类没有属于自己的类型,所以引用它的变量一定是父类或接口,而通过父类或接口类型的变量无法访问子类型中的方法。当然,可以定义一些内部调用的方法供其他继承自父类的方法调用。

  下面是匿名类的语法:

new 父类/接口(参数列表) {类定义};

  匿名类的语法包含以下几部分:

  • new操作符;
  • 匿名类实现的接口或者继承的类;
  • 包含了构造器参数的小括号。注意,当匿名类实现了某个接口时,由于接口没有构造器,因此使用一个空的小括号来表示;
  • 匿名类的定义体。

  就像局部类一样,匿名类也可以捕获变量。下面是几条规则:

  • 匿名类可以访问它的外部类的成员;
  • 匿名类不能访问外部范围中没有使用final修饰或不是近似final的局部变量。
  • 匿名类中会屏蔽外部范围中同名的类型。

  匿名类在成员的定义上和局部类有相同的规则:

  • 不能在匿名类中声明静态初始化器或成员接口;
  • 匿名类中可以有静态成员,前提是这个静态成员必须是常变量。

  可以在匿名类中声明以下元素:

  • 域;
  • 额外的方法(接口中没有定义的方法);
  • 非静态初始化块;
  • 局部类。

  不能在匿名类中定义构造方法。

Java基础教程(14)--嵌套类的更多相关文章

  1. Java基础教程(19)--Object类

      Object类位于类结构树的最顶端,所有的类都是它的直接或间接子类,因此所有的类都继承了Object类的方法,我们可以在需要的时候覆盖这些方法.下面是一些将会在本文中讨论的Object类的方法: ...

  2. java基础之 内部类 & 嵌套类

    参考文档: 内部类的应用场景 http://blog.csdn.net/hivon/article/details/606312 http://wwty.iteye.com/blog/338628 定 ...

  3. Java基础教程——Random随机数类

    Random类 java.util.Random类用于产生随机数.需要导入包: import java.util.Random; 方法 解释 Random() 创建一个Random类对象 Random ...

  4. Java基础教程(12)--深入理解类

    一.方法的返回值   当我们在程序中调用方法时,虚拟机将会跳转到对应的方法中去执行.当以下几种情况发生时,虚拟机将会回到调用方法的语句并继续向下执行: 执行完方法中所有的语句: 遇到return语句: ...

  5. Java基础教程(18)--继承

    一.继承的概念   继承是面向对象中一个非常重要的概念,使用继承可以从逻辑和层次上更好地组织代码,大大提高代码的复用性.在Java中,继承可以使得子类具有父类的属性和方法或者重新定义.追加属性和方法. ...

  6. Java基础-集合的嵌套

    Java基础-集合的嵌套 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.静态导入 静态导入是在JDK1.5后的新特性,可以减少开发的代码量,但是实际用处是很一般,静态导入的标准 ...

  7. Java基础教程:内部类

    Java基础教程:内部类 内部类 内部类,是指在一个类的内部定义的类.就像下面这样: public class EnclosingClass {   . . .   public class Nest ...

  8. Java基础教程:注解

    Java基础教程:注解 本篇文章参考的相关资料链接: 维基百科:https://zh.wikipedia.org/wiki/Java%E6%B3%A8%E8%A7%A3 注解基础与高级应用:http: ...

  9. Java基础教程:网络编程

    Java基础教程:网络编程 基础 Socket与ServerSocket Socket又称"套接字",网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个s ...

随机推荐

  1. 如何在centos下配置redis开机自启动

    2014/11/10补充:其实在redis的下载包中就包含了官方自带的启动脚本,路径在/redis-stable/utils/redis_init_script.在utils目录下也有安装redis- ...

  2. C#穿透session隔离———Windows服务启动UI交互程序

    在Windows服务里面启动其他具有界面的应用程序,需要穿透session隔离,尝试了很多种方法,都可行,现在一一列举下来,并写下几个需要注意的地方. 需要注意的地方 首先要将服务的Account属性 ...

  3. Spring boot创建定时任务

    基于spring boot的应用创建定时任务不要太简单,给一个类加上@Configuration @EnableScheduling注解,然后给该类需要定时执行的方法加上@Scheduled(cron ...

  4. DateTimeField如何自动设置为当前时间并且能被修改 ——django日期时间字段的使用

    参考于:https://www.cnblogs.com/huchong/p/7895263.html 创建django的model时,有DateTimeField.DateField和TimeFiel ...

  5. Python廖雪峰学习笔记——单元测试

    定义:对一个模块.一个类.一个函数进行进行正确性检验的测试性工作.当我们对函数或者模块等进行修改时,单元测试就显得尤为重要. 单元测试 = 测试用例(用来测试的数据)+测试模块

  6. Flask系列03--Flask的路由 app.route中的参数, 动态参数路由

    Flask–路由 添加路由的两种方式 第一种 @app.route("/my_de") def detail() 第二种(了解即可) app.add_url_rule(" ...

  7. 《Python绝技:运用Python成为顶级黑客》 用Python分析网络流量

    1.IP流量将何去何从?——用Python回答: 使用PyGeoIP关联IP地址和物理地址: 需要下载安装pygeoip,可以pip install pygeoip 或者到Github上下载安装htt ...

  8. (1)Oracle基础--用户与登录

    一.用户 · 系统用户 <1> sys,system  sys和system是权限比较高的用户,且sys比system权限高.使用sys登录必须使用sysdba或者sysoper的权限,而 ...

  9. Elasticsearch批量操作API用法介绍

    Elasticsearch的Bulk API允许批量提交index和delete请求,有如下两种用法: 用法1 BulkRequestBuilder requestBuilder = client.p ...

  10. 【OpenCV3】cvRound()、cvFloor()、cvCeil()函数详解

    函数cvRound().cvFloor().cvCeil()都是按照一种舍入方式将浮点型数据转换为整型数据. cvRound():返回跟参数最接近的整数值,即四舍五入: cvFloor()  :返回不 ...