JAVA基础——内部类详解

JAVA内部类详解




  在我的另一篇java三大特性的封装中讲到java内部类的简单概要,这里将详细深入了解java内部类的使用和应用。

  我们知道内部类可分为以下几种:

  • 成员内部类
  • 静态内部类
  • 方法内部类
  • 匿名内部类

  这里我们先将以这个分类来详细了解各个内部类的情况。然后给内部类作出总结。


一、成员内部类

  内部类中最常见的就是成员内部类,也称为普通内部类。我们来看如下代码:

  

  运行结果为:

  

  从上面的代码中我们可以看到,成员内部类的使用方法

  1、 Inner 类定义在 Outer 类的内部相当于 Outer 类的一个成员变量的位置,Inner 类可以使用任意访问控制符,如 public 、 protected 、 private 等

  2、 Inner 类中定义的 test() 方法可以直接访问 Outer 类中的数据,而不受访问控制符的影响,如直接访问 Outer 类中的私有属性a

  3、 定义了成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去 new 一个内部类对象,即:内部类 对象名 = 外部类对象.new 内部类( );

  4、 编译上面的程序后,会发现产生了两个 .class 文件

  

  其中,第二个是外部类的 .class 文件,第一个是内部类的 .class 文件,即成员内部类的 .class 文件总是这样:外部类名$内部类名.class

  另外,友情提示哦:

  1、 外部类是不能直接使用内部类的成员和方法滴。如:

  

  那么外部类如何使用内部类的成员和方法呢??

  答:可先创建内部类的对象,然后通过内部类的对象来访问其成员变量和方法。

  Java 编译器在创建内部类对象时,隐式的把其外部类对象的引用也传了进去并一直保存着。这样就使得内部类对象始终可以访问其外部类对象,同时这也是为什么在外部 类作用范围之外向要创建内部类对象必须先创建其外部类对象的原因。

  2、 如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部类的成员变量,可以使用 this 关键字。如:

  

  运行结果:


二、静态内部类

  静态内部类是 static 修饰的内部类,这种内部类的特点是:

  1、 静态内部类不能直接访问外部类的非静态成员,但可以通过 new 外部类().成员 的方式访问。

  2、 如果外部类的静态成员与内部类的成员名称相同,可通过“类名.静态成员”访问外部类的静态成员;如果外部类的静态成员与内部类的成员名称不相同,则可通过“成员名”直接调用外部类的静态成员。

  3、 创建静态内部类的对象时,不需要外部类的对象,可以直接创建 内部类 对象名= new 内部类();

  

  运行结果 : 


三、方法内部类

  方法内部类就是内部类定义在外部类的方法中,方法内部类只在该方法的内部可见,即只在该方法内可以使用

  

  一定要注意哦:由于方法内部类不能在外部类的方法以外的地方使用,因此方法内部类不能使用访问控制符和 static 修饰符。


四、匿名内部类

  匿名类是不能有名称的类,所以没办法引用他们。必须在创建时,作为new语句的一部分来声明他们。

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

  这就要采用另一种形式 的new语句,如下所示:

   new <类或接口> <类的主体>

这种形式的new语句声明一个 新的匿名类,他对一个给定的类进行扩展,或实现一个给定的接口。他还创建那个类的一个新实例,并把他作为语句的结果而返回。要扩展的类和要实现的接口是
new语句的操作数,后跟匿名类的主体。

注意匿名类的声明是在编译时进行的,实例化在运行时进行。这意味着
for循环中的一个new语句会创建相同匿名类的几个实例,而不是创建几个不同匿名类的一个实例。

从技术上说,匿名类可被视为非静态的内 部类,所以他们具备和方法内部声明的非静态内部类相同的权限和限制。

假如要执行的任务需要一个对象,但却不值得创建全新的对象(原因可能 是所需的类过于简单,或是由于他只在一个方法内部使用),匿名类就显得很有用。匿名类尤其适合在Swing应用程式中快速创建事件处理程式。以下是一个匿名内部类的实例:

  1、匿名内部类的基本实现:

    

  运行结果: 

  可以看到,我们直接将抽象类Person中的方法在大括号中实现了,这样便可以省略一个类的书写,并且,匿名内部类还能用于接口上。

  2、在接口上使用匿名内部类:

  

  运行结果:

  由上面的例子可以看出,只要一个类是抽象的或是一个接口,那么其子类中的方法都可以使用匿名内部类来实现。

  在使用匿名内部类的过程中,我们需要注意如下几点:

      1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。

      2、匿名内部类中是不能定义构造函数的。

      3、匿名内部类中不能存在任何的静态成员变量和静态方法。

      4、匿名内部类为局部内部类(即方法内部类),所以局部内部类的所有限制同样对匿名内部类生效。

      5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

  匿名内部类重点难点

  1. 如果是在一个方法的匿名内部类,可以利用这个方法传进你想要的参数,不过记住,这些参数必须被声明为 final 。

   使用的形参为何要为final??

    参考博文:http://android.blog.51cto.com/268543/384844 http://blog.csdn.net/chenssy/article/details/13170015

   我们给匿名内部类传递参数的时候,若该形参在内部类中需要被使用,那么该形参必须要为final。也就是说:当所在的方法的形参需要被内部类里面使用时,该形参必须为final。

  首先我们知道在内部类编译成功后,它会产生一个class文件,该class文件与外部类并不是同一class文件,仅仅只保留对外部类的引用。当外部类传入的参数需要被内部类调用时,从java程序的角度来看是直接被调用:

    public class OuterClass {
        public void display(final String name,String age){
            class InnerClass{
                void display(){
                    System.out.println(name);
                }
            }
        }
    }  

  从上面代码中看好像name参数应该是被内部类直接调用?其实不然,在java编译之后实际的操作如下:

    public class OuterClass$InnerClass {
        public InnerClass(String name,String age){
            this.InnerClass$name = name;
            this.InnerClass$age = age;
        }  

        public void display(){
            System.out.println(this.InnerClass$name + "----" + this.InnerClass$age );
        }
    }  

    所以从上面代码来看,内部类并不是直接调用方法传递的参数,而是利用自身的构造器对传入的参数进行备份,自己内部方法调用的实际上时自己的属性而不是外部方法传递进来的参数。

    直到这里还没有解释为什么是final。在内部类中的属性和外部方法的参数两者从外表上看是同一个东西,但实际上却不是,所以他们两者是可以任意变化的,也就是说在内部类中我对属性的改变并不会影响到外部的形参,而然这从程序员的角度来看这是不可行的,毕竟站在程序的角度来看这两个根本就是同一个,如果内部类该变了,而外部方法的形参却没有改变这是难以理解和不可接受的,所以为了保持参数的一致性,就规定使用final来避免形参的不改变。

   简单理解就是,拷贝引用,为了避免引用值发生改变,例如被外部类的方法修改等,而导致内部类得到的值不一致,于是用final来让该引用不可改变。

        故如果定义了一个匿名内部类,并且希望它使用一个其外部定义的参数,那么编译器会要求该参数引用是final的。

  2. 匿名内部类中使用初始化代码块

   我们一般都是利用构造器来完成某个实例的初始化工作的,但是匿名内部类是没有构造器的!那怎么来初始化匿名内部类呢?使用构造代码块!利用构造代码块能够达到为匿名内部类创建一个构造器的效果。

    public class OutClass {
        public InnerClass getInnerClass(final int age,final String name){
            return new InnerClass() {
                int age_ ;
                String name_;
                //构造代码块完成初始化工作
                {
                    if(0 < age && age < 200){
                        age_ = age;
                        name_ = name;
                    }
                }
                public String getName() {
                    return name_;
                }  

                public int getAge() {
                    return age_;
                }
            };
        }  

        public static void main(String[] args) {
            OutClass out = new OutClass();  

            InnerClass inner_1 = out.getInnerClass(201, "chenssy");
            System.out.println(inner_1.getName());  

            InnerClass inner_2 = out.getInnerClass(23, "chenssy");
            System.out.println(inner_2.getName());
        }
    }  

五、内部类总结

  学习了上面四种类型的内部类,我们知道了如何使用各个内部类,那么为什么要使用内部类呢??

  参考博文:http://blog.csdn.net/yu422560654/article/details/7466260

  首先举一个简单的例子,如果你想实现一个接口,但是这个接口中的一个方法和你构想的这个类中的一个 方法的名称,参数相同,你应该怎么办?这时候,你可以建一个内部类实现这个接口。由于内部类对外部类的所有内容都是可访问的,所以这样做可以完成所有你直 接实现这个接口的功能。

不过你可能要质疑,更改一下方法的不就行了吗?的确,以此作为设计内部类的理由,实在没有说服 力。

真正的原因是这样的,java中的内部类和接口加在一起,可以的解决常被C++程序员抱怨java中存在的一个问题——没有多继承。实际上,C++的多继承设计起来很复杂,而java通过内部类加上接口,可以很好的实现多继承的效果。
 
内部类:一个内部类的定义是定义在另一个内部的类。

  原因是:

  1.一个内部类的对象能够访问创建它的对象的实现,包括私有数据。

  2.对于同一个包中的其他类来说,内部类能够隐藏起来。

  3.匿名内部类可以很方便的定义回调。

  4.使用内部类可以非常方便的编写事件驱动程序。

内部类可以让你更优雅地设计你的程序结构。下面从以下几个方面来介绍:

  首先看这个例子:

1     public interface Contents {
2      int value();
3     }
4
5     public interface Destination {
6      String readLabel();
7     }  
 1     public class Goods {
 2       private valueRate=2;
 3
 4      private class Content implements Contents {
 5        private int i = 11 * valueRate;
 6       public int value() {
 7        return i;
 8       }
 9      }
10
11      protected class GDestination implements Destination {
12       private String label;
13       private GDestination(String whereTo) {
14        label = whereTo;
15       }
16       public String readLabel() {
17        return label;
18       }
19      }
20
21      public Destination dest(String s) {
22       return new GDestination(s);
23      }
24
25       public Contents cont() {
26       return new Content();
27      }
28     }  

    在这个例子里类 Content 和 GDestination 被定义在了类 Goods 内部,并且分别有着 protected 和 private 修饰符来控制访问级别。Content 代表着 Goods 的内容,而 GDestination 代表着 Goods 的目的地。它们分别实现了两个接口Content和Destination。在后面的main方法里,直接用 Contents c 和 Destination d进行操作,你甚至连这两个内部类的名字都没有看见!这样,内部类的第一个好处就体现出来了——隐藏你不想让别人知道的操作,也即封装性。

  非静态内部类对象有着指向其外部类对象的引用

   修改上面的例子:

 1     public class Goods {
 2       private valueRate=2;
 3
 4      private class Content implements Contents {
 5        private int i = 11 * valueRate;
 6       public int value() {
 7        return i;
 8       }
 9      }
10
11      protected class GDestination implements Destination {
12       private String label;
13       private GDestination(String whereTo) {
14        label = whereTo;
15       }
16       public String readLabel() {
17        return label;
18       }
19      }
20
21      public Destination dest(String s) {
22       return new GDestination(s);
23      }
24
25       public Contents cont() {
26       return new Content();
27      }
28     }  

  在这里我们给 Goods 类增加了一个 private 成员变量 valueRate,意义是货物的价值系数,在内部类 Content 的方法 value() 计算价值时把它乘上。我们发现,value() 可以访问 valueRate,这也是内部类的第二个好处——一个内部类对象可以访问创建它的外部类对象的内容,甚至包括私有变量!这是一个非常有用的特性,为我们在设计时提供了更多的思路和捷径。要想实现这个功能,内部类对象就必须有指向外部类对象的引用。Java 编译器在创建内部类对象时,隐式的把其外部类对象的引用也传了进去并一直保存着。这样就使得内部类对象始终可以访问其外部类对象,同时这也是为什么在外部 类作用范围之外向要创建内部类对象必须先创建其外部类对象的原因。



  个人想说的话:对于java的内部类我花了比较长的时间在网上搜集资料,这部分的学习也有很多理解不够的地方,如果有疑问可以在下面留言,我尽我所能解答;如果有改进的批评的地方欢迎指正,谢谢!

时间: 2024-12-29 10:29:37

JAVA基础——内部类详解的相关文章

JAVA基础——异常详解

阅读目录 一.异常简介 二.try-catch-finally语句 三.throw和throws关键字 四.java中的异常链 五.结束语 JAVA异常与异常处理详解 回到顶部 一.异常简介 什么是异常? 异常就是有异于常态,和正常情况不一样,有错误出错.在java中,阻止当前方法或作用域的情况,称之为异常. java中异常的体系是怎么样的呢? 1.Java中的所有不正常类都继承于Throwable类.Throwable主要包括两个大类,一个是Error类,另一个是Exception类: 2.其

JAVA基础——枚举详解

前言: 在第一次学习面向对象编程时,我记得最深的一句话就是"万物皆对象".于是我一直秉承着这个思想努力的学习着JAVA,直到学习到枚举(Enum)时,看着它颇为奇怪的语法--我一直在想,这TM是个什么鬼???当时学习OOP时也是被类啊接口什么的整的有点昏头转向的于是就把这个小细节忽略掉了.后来到了公司工作后慢慢的又需要用上枚举了,看着它一副神秘兮兮的样子我还是决定要好好的深挖一下!以下链接是了解枚举时所参考的博客.如发现本文有错误或知识遗漏欢迎在评论中指正! 反编译那些事儿(二)-枚举

Java 基础知识详解

由于有C#的基础,Java的基础知识基本是略过,这里当做复习一遍吧! Java的三种技术架构: JavaEE:(Java PlatForm Enterprise Edition) Java开发企业级的应用,主要针对Web JavaSE;(Java PlatForm  Standard Edition) 完成桌面程序的开发,是其他两个的基础 JavaME:(Java PlatForm Micro Edition)开发电子消费产品和嵌入式设备,如Android Java数据类型 (1)基本数据类型(

Java 之 -------------- 内部类 详解

java中的内部类,分为 内部类,局部内部类,匿名内部类! 内部类: 顾名思义 定义在类的内部的类叫做 内部类! 局部内部类: 定义在局部函数中的 类,叫做 局部内部类! 匿名内部类:  没有名字的内部类! 玩 android  都知道,android  比较常见 匿名内部类! 内部类的优点: 1 内部类可以直接 访问 外部类的所有成员! 因为,当你调用外部类的时候,却生了 outer.this.成员变量或函数! 2 含有静态成员的内部类, 必须是静态的! 3 内部类一般用在设计中! 弊端: 1

java(24) - 内部类详解

一.内部类:        1).内部类的定义:通俗的讲就是在一个类的里面再定义一个类. 2).内部类可以public,private,protected等修饰,也可以是静态static. 二.四种内部类的实现: 1).静态内部类: 例子: class InnerClass { public static int count = 10; //静态内部类 public static class Inner{ public void print(){ System.out.println(count

Java中内部类详解—匿名内部类

什么是内部类? 将一个类A定义在另一个类B里面,里面的那个类A就称为内部类,B则称为外部类.   成员内部类 定义在类中方法外的类. 定义格式: class 外部类 { class 内部类{ } } 在描述事物时,若一个事物内部还包含其他事物,就可以使用内部类这种结构.比如,汽车类Car 中包含发动机类Engine ,这时,Engine就可以使用内部类来描述,定义在成员位置. 代码举例: class Car { //外部类 class Engine { //内部类 } } 访问特点 成员内部类可

[转] Java内部类详解

作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本博客中未标明转载的文章归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利. Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就来一探究竟.下面是本文的目录大纲: 一.内部类基础 二.

Java内部类详解(一)(转自:http://blog.csdn.net/wangpeng047/article/details/12344593)

很多人对于Java内部类(Inner Class)都十分陌生,甚至听都没听过也没有使用过,内部类在Java中其实是比较重要的一块内容,掌握好这门知识对于编程来说,犹如插上一对翅膀. 一.概念 内部类是指在一个外部类的内部再定义一个类,类名不需要和文件名相同. 对于一个名为outer的外部类和其内部定义的名为inner的内部类.编译完成后会生成outer.class和outer$inner.class两个类.所以内部类的成员变量.方法名可以和外部类的相同. 内部类可以是静态static和非静态的,

Java内部类详解 2

Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就来一探究竟.下面是本文的目录大纲: 一.内部类基础 二.深入理解内部类 三.内部类的使用场景和好处 四.常见的与内部类相关的笔试面试题 若有不正之处,请多谅解并欢迎批评指正. 请尊重作者劳动成果,转载请标明原文链接: http://www.cnblogs.com/dolphin0520/p/3811