使用内部类的原因:

(1)可以访问该类定义所在作用域中的数据,包括私有数据。

(2)可以对同一个包中的其它类隐藏起来。

(3)当想要定义一个回调函数且不想编写大量代码时,使用匿名内部类比较便捷。

下面主要分几种情况进行讨论:

(a)使用内部类访问对象状态

class TalkingClock {
	private int interval;
	private boolean beep;

	public TalkingClock(int interval, boolean beep) {
		this.interval = interval;
		this.beep = beep;
	}

	public void start() {
		ActionListener listener = new TimePrinter();
		Timer t = new Timer(interval, listener);
		t.start();
	}

	/** 内部类 **/
	/**将TimePrinter声明为私有,只有TalkingClock能够生成**/
	private class TimePrinter implements ActionListener {
		@Override
		public void actionPerformed(ActionEvent e) {
			if (beep) {
				Toolkit.getDefaultToolkit().beep();
			}
		}
	}
}

为了能够让程序正常运行,内部类的对象总有一个隐式引用,它指向了创建它的外部类对象。

这个引用在内部类的定义中是不可见的。为了进一步说明,将外围类对象的引用称为outer。actionPerformed方法等价于下列形式:

public void actionPerformed(ActionEvent e) {
	if (outer.beep) {
	      Toolkit.getDefaultToolkit().beep();
	}
}

外围类的引用在构造器中设置。,编译器修改了所有内部类的构造器,并添加一个外围类引用的参数。

由于TimePrinter没有构造器,编译器会为其生成一个默认构造器(outer在此仅用来说明内部类的机制):

public TimePrinter(TalkingClock clock){
 	  outer = clock
}

当在start方法中创建了TimePrinter对象后,编译器会将this引用传递给当前的构造器:

ActionListener listener = new TimePrinter(this);

(b)局部内部类

在上面的代码中,TimePrinter类只在start方法中创建这个类型的对象时使用了一次。

当遇到类似情形时,可以在方法中定义局部类:

	public void start() {
		class TimePrinter implements ActionListener {
			@Override
			public void actionPerformed(ActionEvent e) {
				if (beep) {
					Toolkit.getDefaultToolkit().beep();
				}
			}
		}
	}

(c)由外部方法访问final变量

局部类内部类不仅能访问包含它们的外部类,还可以访问局部变量。不过,那些局部变量必须被声明为final。

将TalkingClock构造器中的参数interval和beep移到start方法中。

public void start(int interval, final boolean beep) {
		class TimePrinter implements ActionListener {
			@Override
			public void actionPerformed(ActionEvent e) {
				if (beep) {
					Toolkit.getDefaultToolkit().beep();
				}
			}
		}
		ActionListener listener = new TimePrinter();
		Timer t = new Timer(interval, listener);
		t.start();
	}

需要注意的是:TalkingClock类不再需要存储实例变量beep了,它只是引用start方法中的beep参数变量。

程序if(beep)... 在start方法内部,为什么不能访问beep变量的值呢?

内部的控制流程如下:

(1)调用start方法。

(2)调用内部的TimePrinter构造器,以初始化对象变量listener。

(3)将listener引用传递给Timer构造器,定时器打开,start方法结束。

此时start方法的beep参数变量不复存在。

(4)然后,actionPerformed方法执行if(beep)....。

为了能够让actionPerformed方法能够正常工作,TimePrinter类在beep域释放之前将beep域用start方法的局部变量进行了备份。

编译器为局部内部类构造了名为TalkingClock$TimePrinter。

	class TalkingClock$1TimePrinter{
		TalkingClock$1TimePrinter(TalkingClock, boolean);

		public void actionPerformed(java.awt.event.ActionEvent);

		final boolean val$beep;
		final TalkingClock this$0;

	}

注意构造器的boolean参数和val$beep实例变量。当创建一个对象时,beep就会被传递给构造器,并存储在val$beep域中。

编译器必须检测对局部变量的访问,为每一个变量建立相应的数据,并将局部变量拷贝到构造器中。

JAVA内部类_1的更多相关文章

  1. Java内部类final语义实现

    本文描述在java内部类中,经常会引用外部类的变量信息.但是这些变量信息是如何传递给内部类的,在表面上并没有相应的线索.本文从字节码层描述在内部类中是如何实现这些语义的. 本地临时变量 基本类型 fi ...

  2. Java内部类详解

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

  3. 黑马----JAVA内部类

    黑马程序员:Java培训.Android培训.iOS培训..Net培训 黑马程序员--JAVA内部类 一.内部类分为显式内部类和匿名内部类. 二.显式内部类 1.即显式声明的内部类,它有类名. 2.显 ...

  4. java 内部类 *** 最爱那水货

    注: 转载于http://blog.csdn.net/jiangxinyu/article/details/8177326 Java语言允许在类中再定义类,这种在其它类内部定义的类就叫内部类.内部类又 ...

  5. java内部类和匿名内部类

    内部类即是包含在类里面的又一个类. java内部类分为: 成员内部类.静态嵌套类.方法内部类.匿名内部类 . 内部类的共性 (1).内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.clas ...

  6. Java内部类小程序(成员内部类,静态内部类,匿名内部类)

    /** * 测试java内部类(成员内部类,静态内部类,匿名内部类) * 局部内部类不常用,就不写了. * @package :java05 * @author shaobn * @Describe ...

  7. [转] Java内部类详解

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

  8. java内部类的作用分析

    提起Java内部类(Inner Class)可能很多人不太熟悉,实际上类似的概念在C++里也有,那就是嵌套类(Nested Class),关于这两者的区别与联系,在下文中会有对比.内部类从表面上看,就 ...

  9. 9)Java内部类(Inner Class)

      内部类:不可以有静态数据,静态方法或者又一个静态内部类      内部类的优点:隐藏类的细节,内部类可以声明为私有.内部类可以访问外部类的对象(包括private) 静态内部类:可以有静态数据,静 ...

随机推荐

  1. Eclipse插件安装4种方法

    第一种:直接复制法 假设Eclipse的安装目录在C:\eclipse,解压下载的eclipse 插件或者安装eclipse 插件到指定目录AA(如:c:\AA)文件夹,打开AA 文件夹,在AA文件夹 ...

  2. node的异常处理

    Node是单线程运行环境,一旦抛出的异常没有被捕获,就会引起整个进程的崩溃.所以,Node的异常处理对于保证系统的稳定运行非常重要. node的处理方法: 1.使用throw语句抛出异常 常用的捕获异 ...

  3. Java并发中的CopyOnWrite容器

    Copy-On-Write简称COW,是一种用于程序设计中的优化策略.其基本思路是,从一开始大家都在共享同一个内容,当某个人想要修改这个内容的时候,才会真正把内容Copy出去形成一个新的内容然后再改, ...

  4. 77. Combinations(medium, backtrack, 重要, 弄了1小时)

    Given two integers n and k, return all possible combinations of k numbers out of 1 ... n. For exampl ...

  5. PHP 5 Array 函数

    PHP Array 简介 PHP Array 函数允许您访问并操作数组. 支持简单的数组和多维数组. 安装 PHP Array 函数是 PHP 核心的组成部分.无需安装即可使用这些函数. PHP 5 ...

  6. TensorFlow入门和示例分析

    本文以TensorFlow源码中自带的手写数字识别Example为例,引出TensorFlow中的几个主要概念.并结合Example源码一步步分析该模型的实现过程. 一.什么是TensorFlow 在 ...

  7. 排查Full GC

    我们的Java应用因频繁FULL GC导致性能降低很多,经过多人的定位也没有结论,于是我自主请命,经过一天的研究终于搞定了,现把经验与大家共享,相关的gc日志如下: 4.758: [Full GC [ ...

  8. proc文件系统探索 之 根目录下的文件[1]

    2.1根目录下的文件2.1.1lock文件内核锁,记录与被打开的文件有关的锁信息. 该文件显示当前被内核锁定的文件.该文件包含的内容是内核调试数据,根据使用的系统的这些数据会变化很大.一个/proc/ ...

  9. Hadoop 伪分布式安装、运行测试例子

    1. 配置linux系统环境 centos 6.4 下载地址:http://pan.baidu.com/s/1geoSWuv[VMWare专用CentOS.rar](安装打包好的VM压缩包) 并配置虚 ...

  10. Kafka学习笔记1:概念

    一.简介 Apache Kafka是一个分布式的消息系统,作为一个分布式的日志提交服务. Kafka 是一个分布式的.可分区的.可复制的日志提交服务. 它提供了功能性的消息系统,有它自己独特的设计. ...