Java知多少(37)静态内部类、匿名内部类、成员式内部类和局部内部类

内部类可以是静态(static)的,可以使用 public、protected 和 private 访问控制符,而外部类只能使用 public,或者默认。

成员式内部类

在外部类内部直接定义(不在方法内部或代码块内部)的类就是成员式内部类,它可以直接使用外部类的所有变量和方法,即使是 private 的。外部类要想访问内部类的成员变量和方法,则需要通过内部类的对象来获取。

请看下面的代码:

 1 public class Outer{
 2     private int size;
 3     public class Inner {
 4         public void dostuff() {
 5             size++;
 6         }
 7     }
 8     public void testTheInner() {
 9         Inner in = new Inner();
10         in.dostuff();
11     }
12 }

成员式内部类如同外部类的一个普通成员。

成员式内部类可以使用各种修饰符,包括 public、protected、private、static、final 和 abstract,也可以不写。

若有 static 修饰符,就为类级,否则为对象级。类级可以通过外部类直接访问,对象级需要先生成外部的对象后才能访问。

非静态内部类中不能声明任何 static 成员。

内部类可以相互调用,例如:

成员式内部类的访问

1 class A {
2     // B、C 间可以互相调用
3     class B {}
4     class C {}
5 }

内部类的对象以成员变量的方式记录其所依赖的外层类对象的引用,因而可以找到该外层类对象并访问其成员。该成员变量是系统自动为非 static 的内部类添加的,名称约定为“outClassName.this”。

1) 使用内部类中定义的非静态变量和方法时,要先创建外部类的对象,再由“outObjectName.new”操作符创建内部类的对象,再调用内部类的方法,如下所示:

 1 public class Demo{
 2     public static void main(String[] args) {
 3         Outer outer = new Outer();
 4         Outer.Inner inner = outer.new Inner();
 5         inner.dostuff();
 6     }
 7 }
 8 class Outer{
 9     private int size;
10     class Inner{
11         public void dostuff() {
12             size++;
13         }
14     }
15 }

2) static 内部类相当于其外部类的 static 成员,它的对象与外部类对象间不存在依赖关系,因此可直接创建。示例如下:

 1 public class Demo{
 2     public static void main(String[] args) {
 3         Outer.Inner inner = new Outer.Inner();
 4         inner.dostuff();
 5     }
 6 }
 7 class Outer{
 8     private static int size;
 9     static class Inner {
10         public void dostuff() {
11             size++;
12             System.out.println("size=" + size);
13         }
14     }
15 }

运行结果:
size=1

3) 由于内部类可以直接访问其外部类的成分,因此当内部类与其外部类中存在同名属性或方法时,也将导致命名冲突。所以在多层调用时要指明,如下所示:

 1 public class Outer{
 2     private int size;
 3     public class Inner{
 4         private int size;
 5         public void dostuff(int size){
 6             size++;  // 局部变量 size;
 7             this.size;  // 内部类的 size
 8             Outer.this.size++;  // 外部类的 size
 9         }
10     }
11 }

局部内部类

局部内部类(Local class)是定义在代码块中的类。它们只在定义它们的代码块中是可见的。

局部类有几个重要特性:

  1. 仅在定义了它们的代码块中是可见的;
  2. 可以使用定义它们的代码块中的任何局部 final 变量;
  3. 局部类不可以是 static 的,里边也不能定义 static 成员;
  4. 局部类不可以用 public、private、protected 修饰,只能使用缺省的;
  5. 局部类可以是 abstract 的。

请看下面的代码:

 1 public class Outer {
 2     public static final int TOTAL_NUMBER = 5;
 3     public int id = 123;
 4     public void func() {
 5         final int age = 15;
 6         String str = "http://www.weixueyuan.net";
 7         class Inner {
 8             public void innerTest() {
 9                 System.out.println(TOTAL_NUMBER);
10                 System.out.println(id);
11                 // System.out.println(str);不合法,只能访问本地方法的final变量
12                 System.out.println(age);
13             }
14         }
15         new Inner().innerTest();
16     }
17     public static void main(String[] args) {
18         Outer outer = new Outer();
19         outer.func();
20     }
21 }

运行结果:
5
123
15

匿名内部类

匿名内部类是局部内部类的一种特殊形式,也就是没有变量名指向这个类的实例,而且具体的类实现会写在这个内部类里面。

注意:匿名类必须继承一个父类或实现一个接口。

不使用匿名内部类来实现抽象方法:

 1 abstract class Person {
 2     public abstract void eat();
 3 }
 4 class Child extends Person {
 5     public void eat() {
 6         System.out.println("eat something");
 7     }
 8 }
 9 public class Demo {
10     public static void main(String[] args) {
11         Person p = new Child();
12         p.eat();
13     }
14 }

运行结果:
eat something

可以看到,我们用Child继承了Person类,然后实现了Child的一个实例,将其向上转型为Person类的引用。但是,如果此处的Child类只使用一次,那么将其编写为独立的一个类岂不是很麻烦?

这个时候就引入了匿名内部类。使用匿名内部类实现:

 1 abstract class Person {
 2     public abstract void eat();
 3 }
 4 public class Demo {
 5     public static void main(String[] args){
 6
 7         // 继承 Person 类
 8         new Person() {
 9             public void eat() {
10                 System.out.println("eat something");
11             }
12         }.eat();
13     }
14 }

可以看到,匿名类继承了 Person 类并在大括号中实现了抽象类的方法。

内部类的语法比较复杂,实际开发中也较少用到,本教程不打算进行深入讲解,各位读者也不应该将内部类作为学习Java的重点。

系列文章:

Java知多少(1)语言概述

Java知多少(2)虚拟机(JVM)以及跨平台原理

Java知多少(3) 就业方向

Java知多少(4)J2SE、J2EE、J2ME的区别

Java知多少(5) Java开发环境的搭建

Java知多少(6)第一个程序示例

Java知多少(7)类与对象

Java知多少(8)类库及其组织结构

Java知多少(9) import及Java类的搜索路径

Java知多少(10)数据类型及变量

Java知多少(11)数据类型转换

Java知多少(12)运算符

Java知多少(13)流程控制

Java知多少(14)数组

Java知多少(15)字符串

Java知多少(16)StringBuffer与StringBuider

Java知多少(17)强调一下编程风格

Java知多少(18)类的定义及其实例化

Java知多少(19)访问修饰符(访问控制符)

Java知多少(20)变量的作用域

Java知多少(21)this关键字详解

Java知多少(22)方法重载

Java知多少(23)类的基本运行顺序

Java知多少(24)包装类、拆箱和装箱详解

Java知多少(25)再谈Java包

Java知多少(26)源文件的声明规则

Java知多少(27)继承的概念与实现

Java知多少(28)super关键字

Java知多少(29)覆盖和重载

Java知多少(30)多态和动态绑定

Java知多少(31)static关键字以及Java静态变量和静态方法

Java知多少(32)instanceof

Java知多少(33)多态对象的类型转换

Java知多少(34)final关键字:阻止继承和多态

Java知多少(35)Object类

Java知多少(36)内部类及其实例化

Java知多少(37)静态内部类、匿名内部类、成员式内部类和局部内部类

时间: 2024-10-14 21:07:27

Java知多少(37)静态内部类、匿名内部类、成员式内部类和局部内部类的相关文章

注意java8中已经没有匿名内部类和局部内部类只能访问final变量的限制了!

今天写Android代码编译器报错Variable 'arrayAdapter' is accessed from within inner class, needs to be declared final. 于是打算找几篇博客学习学习,结果不管我怎么弄,在java编译器上就是不报错啊! 转念一想,是不是我用JDK1.8的原因,百度一下,知乎证实了我的猜想. 具体见:http://docs.oracle.com/javase/tutorial/java/javaOO/localclasses.

内部类练习题(外部类访问内部类成员、内部类访问外部类成员、顶级类访问内部类成员)

package com.Summer_0429.cn; /** * @author Summer * 内部类实例: * 定义一只猫类,猫有: * 1.重量 * 2.猫的身体: * 1)颜色 * 2)显示猫的身体的信息(): * 3.显示猫的整体信息(): * 要求:创建一只小猫,显示它的整体信息. * */ class Cat{ private double weight; public Cat(double weight){ this.weight = weight; } //内部类:成员内部

Java知多少(完结篇)

Java知多少(1)语言概述 Java知多少(2)虚拟机(JVM)以及跨平台原理 Java知多少(3) 就业方向 Java知多少(4)J2SE.J2EE.J2ME的区别 Java知多少(5) Java开发环境的搭建 Java知多少(6)第一个程序示例 Java知多少(7)类与对象 Java知多少(8)类库及其组织结构 Java知多少(9) import及Java类的搜索路径 Java知多少(10)数据类型及变量 Java知多少(11)数据类型转换 Java知多少(12)运算符 Java知多少(1

Java知多少(上)

ava知多少(1)语言概述 Java知多少(2)虚拟机(JVM)以及跨平台原理 Java知多少(3) 就业方向 Java知多少(4)J2SE.J2EE.J2ME的区别 Java知多少(5) Java开发环境的搭建 Java知多少(6)第一个程序示例 Java知多少(7)类与对象 Java知多少(8)类库及其组织结构 Java知多少(9) import及Java类的搜索路径 Java知多少(10)数据类型及变量 Java知多少(11)数据类型转换 Java知多少(12)运算符 Java知多少(13

java:内部类与外部类的区别和联系

注意事项一:在内部类中可以随意使用外部类的成员方法以及成员变量. 众所周知,在定义成员方法或者成员变量的时候,可以给其加上一些权限的修饰词,以防止其他类的访问.如在成员变量或者成员方法前面,加上Private 关键字,则其他类就无法调用这个类中的成员方法或则和成员变量.但是,如果这个类有成员内部类,则不受这方面的限制.也就是说,在成员内部类中可以随意引 用外部类的成员方法以及成员变量,即使这些类成员方法或者成员变量被修饰了private.如在成员外部类中定义了一个i变量,并且利用private关

Java学习(十六)、成员内部类,静态内部类,匿名内部类,局部内部类

定义:将一个类定义在另一个类的内部.内部的类称为内部类; public class Outer{ class inner{ //省略 } } 特点: ①   内部类可以很好的实现隐藏,可以使用protected,private修饰符; ②   内部类可以直接访问外部类的所有成员,包括私有成员; ③   外部类不能直接访问内部类的成员,必须首先要建立内部类的对象才能访问; ④   内部类可以解决一些问题,比如间接的去实现多继承.可以避免修改接口而实现同一个类中两种同名方法的调用; 成员内部类及应用

Java知多少(39)interface接口

在抽象类中,可以包含一个或多个抽象方法:但在接口(interface)中,所有的方法必须都是抽象的,不能有方法体,它比抽象类更加“抽象”. 接口使用 interface 关键字来声明,可以看做是一种特殊的抽象类,可以指定一个类必须做什么,而不是规定它如何去做. 现实中也有很多接口的实例,比如说串口电脑硬盘,Serial ATA委员会指定了Serial ATA 2.0规范,这种规范就是接口.Serial ATA委员会不负责生产硬盘,只是指定通用的规范. 希捷.日立.三星等生产厂家会按照规范生产符合

Java知多少(38)抽象类的概念和使用

在自上而下的继承层次结构中,位于上层的类更具有通用性,甚至可能更加抽象.从某种角度看,祖先类更加通用,它只包含一些最基本的成员,人们只将它作为派生其他类的基类,而不会用来创建对象.甚至,你可以只给出方法的定义而不实现,由子类根据具体需求来具体实现. 这种只给出方法定义而不具体实现的方法被称为抽象方法,抽象方法是没有方法体的,在代码的表达上就是没有“{}”.包含一个或多个抽象方法的类也必须被声明为抽象类. 使用 abstract 修饰符来表示抽象方法和抽象类. 抽象类除了包含抽象方法外,还可以包含

Java知多少(18)类的定义及其实例化

类必须先定义才能使用.类是创建对象的模板,创建对象也叫类的实例化. 下面通过一个简单的例子来理解Java中类的定义: 1 public class Dog{ 2 String name; 3 int age; 4 5 void bark(){ // 汪汪叫 6 System.out.println("汪汪,不要过来"); 7 } 8 9 void hungry(){ // 饥饿 10 System.out.println("主人,我饿了"); 11 } 12 } 对