Java 高级类(下) —— 内部类和匿名类

Java内部类(Inner Class),类似的概念在C++里也有,那就是嵌套类(Nested Class),乍看上去内部类似乎有些多余,它的用处对于初学者来说可能并不是那么显著,但是随着对它的深入了解,你会发现Java的设计者在内部类身上的确是用心良苦。学会使用内部类,是掌握Java高级编程的一部分,它可以让你更优雅地设计你的程序结构。下面从以下几个方面来介绍:

 

第一次见面

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. public interface Contents {  
  2.     int value();  
  3. }  
  4.   
  5. public interface Destination {  
  6.     String readLabel();  
  7. }  
  8.   
  9. public class Goods {  
  10.     private class Content implements Contents {  
  11.         private int i = 11;  
  12.         public int value() {  
  13.             return i;  
  14.         }  
  15.     }  
  16.   
  17.     protected class GDestination implements Destination {  
  18.         private String label;  
  19.         private GDestination(String whereTo) {  
  20.             label = whereTo;  
  21.         }  
  22.         public String readLabel() {  
  23.             return label;  
  24.         }  
  25.     }  
  26.   
  27.     public Destination dest(String s) {  
  28.         return new GDestination(s);  
  29.     }  
  30.     public Contents cont() {  
  31.         return new Content();  
  32.     }  
  33. }  
  34.   
  35. class TestGoods {  
  36.     public static void main(String[] args) {  
  37.         Goods p = new Goods();  
  38.         Contents c = p.cont();  
  39.         Destination d = p.dest("Beijing");  
  40.     }  
  41. }  

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

     同时,我们也发现了在外部类作用范围之外得到内部类对象的第一个方法,那就是利用其外部类的方法创建并返回。上例中的cont()和dest()方法就是这么做的。那么还有没有别的方法呢?当然有,其语法格式如下

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. outerObject = new outerClass(Constructor Parameters);  
  2.   
  3. outerClass.innerClass innerObject = outerObject.new InnerClass(Constructor Parameters);  

      注意在创建非静态内部类对象时,一定要先创建起相应的外部类对象。至于原因,也就引出了我们下一个话题——非静态内部类对象有着指向其外部类对象的引用

 

一、常规内部类

常规内部类没有用static修饰且定义在在外部类类体中。

1.常规内部类中的方法可以直接使用外部类的实例变量和实例方法。

2.在常规内部类中可以直接用内部类创建对象

对刚才的例子稍作修改:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  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.     public Contents cont() {  
  25.         return new Content();  
  26.     }  
  27. }  

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

    有人会问,如果内部类里的一个成员变量与外部类的一个成员变量同名,也即外部类的同名成员变量被屏蔽了,怎么办?没事,Java里用如下格式表达外部类的引用:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. outerClass.this  

有了它,我们就不怕这种屏蔽的情况了。

 

可以再看一个例子:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. package cn.com.qiang.neibulei2;  
  2.   
  3. public class MyOuter{  
  4.     private int x = 100;  
  5.   
  6.      // 创建内部类  
  7.     class MyInner {  
  8.         private String y = "Hello!";  
  9.   
  10.         public void innerMethod(){  
  11.             System.out.println("内部类中 String =" + y);  
  12.             System.out.println("外部类中的x =" + x);// 直接访问外部类中的实例变量x  
  13.             outerMethod();  
  14.   
  15.             System.out.println"调用完外部类方法后,x = "+MyOuter.this.x);   
  16.       
  17.         }  
  18.   
  19.     }  
  20.   
  21.     public void outerMethod() {  
  22.         x++;  
  23.     }  
  24.   
  25.     public void makeInner() {  
  26.         //在外部类方法中创建内部类实例  
  27.         @SuppressWarnings("unused")  
  28.         MyInner in = new MyInner();  
  29.     }  
  30.   
  31.      /** 
  32.       * @param args 
  33.       */  
  34.     public static void main(String[] args) {  
  35.         MyOuter mo = new MyOuter();  
  36.         // 使用外部类构造方法创建mo对象  
  37.         MyOuter.MyInner inner = mo.new MyInner();//常规内部类需要通过外部类的实例才能创建对象,与实例变量需要通过对象来访问相似  
  38.         // 创建inner对象  
  39.         inner.innerMethod();  
  40.         // TODO Auto-generated method stub  
  41.   
  42.     }     
  43. }  

执行结果如下:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. 内部类中 String =Hello!  
  2. 外部类中的x =100  
  3. 调用完外部类方法后,x = 101  

 

二、静态内部类

     和普通的类一样,内部类也可以有静态的。不过和非静态内部类相比,区别就在于静态内部类没有了指向外部的引用。这实际上和C++中的嵌套类很相像了,Java内部类与C++嵌套类最大的不同就在于是否有指向外部的引用这一点上,当然从设计的角度以及以它一些细节来讲还有区别。

     除此之外,在任何非静态内部类中,都不能有静态数据,静态方法或者又一个静态内部类(内部类的嵌套可以不止一层)。不过静态内部类中却可以拥有这一切。这也算是两者的第二个区别吧。

下面看一个例子:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. package cn.com.qiang.neibulei2;  
  2.   
  3. public class MyOuter2 {  
  4.     public static int x=100;  
  5.       
  6.     public static class MyInner{  
  7.         private String y="Hello!";  
  8.           
  9.         public void innerMethod(){  
  10.             System.out.println("我是内部类 x = "+x);  
  11.             System.out.println("我是内部类 y = "+y);  
  12.         }  
  13.     }  
  14.   
  15.      /** 
  16.       * @param args 
  17.       */  
  18.      public static void main(String[] args) {  
  19.          MyOuter2.MyInner si = new MyOuter2.MyInner();//静态内部类不通过外部实例就可以创建对象;与类变量可以通过类名访问相似  
  20.          si.innerMethod();  
  21.       // TODO Auto-generated method stub  
  22.      }  
  23. }   

执行结果如下:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. 我是内部类 x = 100  
  2. 我是内部类 y = Hello!  

可以看到 MyOuter2.MyInner si = new MyOuter2.MyInner(); 中并不需要写成

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. MyOuter2.MyInner si = new MyOuter2.new MyInner();  

 

三、局部内部类

         在方法体或语句块(包括方法、构造方法、局部块或静态初始化块)内部定义的类称为局部内部类

        局部内部类不能加任何访问修饰符,因为它只对局部块有效

1.局部内部类只在方法体中有效,就想定义的局部变量一样,在定义的方法体外不能创建局部内部类的对象

2.在方法内部定义类时,应注意以下问题:

1)、方法定义局部内部类同方法定义局部变量一样,不能使用private、protected、public等访问修饰说明符修饰,也不能使用static修饰,但可以使用final和abstract修饰;
2)、方法中的内部类可以访问外部类成员。对于方法的参数和局部变量,必须有final修饰才可以访问;
3)、static方法中定义的内部类可以访问外部类定义的static成员;

 

下面看一个实例:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. package cn.com.qiang.neibulei2;  
  2.   
  3. public class MyOuter3{  
  4.     private int size = 5,y = 7;  
  5.       
  6.     public Object makeInner(int localVar){  
  7.         final int finalLocalVar = localVar;  
  8.         //创建内部类,该类只在makeInner()方法有效,就像局部变量一样。在方法体外部不能创建MyInner类的对象  
  9.         class MyInner{  
  10.             int y = 4;  
  11.             public String toString(){  
  12.                 return "OuterSize:"+size+  
  13.                 "\n"+"finalLocalVar = "+finalLocalVar+"\n"+"this.y = "+this.y;  
  14.             }  
  15.         }     
  16.         return new MyInner();    
  17.     }  
  18.   
  19.     /** 
  20.      * @param args 
  21.      */  
  22.     public static void main(String[] args) {  
  23.           Object obj = new MyOuter3().makeInner(47);//创建Jubu对象obj,并调用它的makeInner()方法,该方法返回一个  
  24.           //该方法返回一个MyInner类型的的对象obj,然后调用其同toString方法。  
  25.           System.out.println(obj.toString());  
  26.     }  
  27. }    

执行结果如下:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. OuterSize:5  
  2. finalLocalVar = 47  
  3. this.y = 4  

 

四、匿名内部类 

      定义类的最终目的是创建一个类的实例,但是如果某个类的实例只是用一次,则可以将类的定义与类的创建,放到与一起完成,或者说在定义类的同时就创建一个类。以这种方法定义的没有名字的类成为匿名内部类

声明和构造匿名内部类的一般格式如下:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. new interfacename(){......}; 或 new superclassname(){......};  

1.匿名内部类可以继承一个类或实现一个接口,这里的ClassOrInterfaceName是匿名内部类所继承的类名或实现的接口名。但匿名内部类不能同时实现一个接口和继承一个类,也不能实现多个接口。如果实现了一个接口,该类是Object类的直接子类,匿名类继承一个类或实现一个接口,不需要extends和implements关键字。

2.由于匿名内部类没有名称,所以类体中不能定义构造方法,由于不知道类名也不能使用关键字来创建该类的实例。实际上匿名内部类的定义、构造、和第一次使用都发生在同样一个地方。此外,上式是一个表达式,返回的是一个对象的引用,所以可以直接使用或将其复制给一个对象变量。例:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. TypeName obj = new Name(){  
  2.   
  3.    /*此处为类体*/  
  4. }  
  5. //同样,也可以将构造的对象作为调用的参数。例:  
  6. someMethod(new Name(){  
  7.    /*此处为类体*/  });  

 

下面接着举例子:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. package cn.com.qiang.neibulei2;  
  2.   
  3. public class MyOuter4 {  
  4.     private int size=5;  
  5.   
  6.     public Object makeInner(int localVar){  
  7.         final int finalLocalVar = localVar;  
  8.         return new Object(){  
  9.            //使用匿名内部类  
  10.             public String toString(){  
  11.                 return "OuterSize = "+size+"\nfinalLocalVar = "+finalLocalVar;  
  12.             }  
  13.         };  
  14.     }  
  15.   
  16.      /** 
  17.       * @param args 
  18.       */  
  19.     public static void main(String args[]){  
  20.         Object obj = new MyOuter4().makeInner(47);  
  21.         System.out.println(obj.toString());  
  22.     }  
  23. }  

执行结果如下:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. OuterSize = 5  
  2. finalLocalVar = 47  

 

在java的事件处理的匿名适配器中,匿名内部类被大量的使用。例如在想关闭窗口时加上这样一句代码:

[java] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. frame.addWindowListener(new WindowAdapter(){  
  2.      public void windowClosing(WindowEvent e){  
  3.           System.exit(0);  
  4.      }  
  5. });  

 

有一点需要注意的是,匿名内部类由于没有名字,所以它没有构造函数(但是如果这个匿名内部类继承了一个只含有带参数构造函数的父类,创建它的时候必须带上这些参数,并在实现的过程中使用super关键字调用相应的内容)。如果你想要初始化它的成员变量,有下面几种方法:

1)如果是在一个方法的匿名内部类,可以利用这个方法传进你想要的参数,不过记住,这些参数必须被声明为final。 
2)将匿名内部类改造成有名字的局部内部类,这样它就可以拥有构造函数了。 
3)在这个匿名内部类中使用初始化代码块。

 

为什么需要内部类?

java内部类有什么好处?为什么需要内部类?

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

    不过你可能要质疑,更改一下方法的不就行了吗?

    的确,以此作为设计内部类的理由,实在没有说服力。

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

 

 java内部类总结

1、 在方法间定义的非静态内部类:

1)外围类和内部类可互相访问自己的私有成员。

2)内部类中不能定义静态成员变量。
      在外部类作用范围之外向要创建内部类对象必须先创建其外部类对象

2、 在方法间定义的静态内部类:

1)只能访问外部类的静态成员。
       静态内部类没有了指向外部的引用

3、在方法中定义的局部内部类:

1)该内部类没有任何的访问控制权限
2)外围类看不见方法中的局部内部类的,但是局部内部类可以访问外围类的任何成员。
3)方法体中可以访问局部内部类,但是访问语句必须在定义局部内部类之后。
4)局部内部类只能访问方法体中的常量,即用final修饰的成员。


4、在方法中定义的匿名内部类:

1)没有构造器,取而代之的是将构造器参数传递给超类构造器

      当你只需要创建一个类的对象而且用不上它的名字时,使用匿名内部类可以使代码看上去简洁清楚。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/401997.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java 高级类(上) —— 抽象类和接口

在面向对象的概念中,我们知道所有的对象都是通过类来描绘的,但是并不是所有的类都是用来描绘对象的,如果一个类中没有包含足够的信息来描绘一个具体的对象,这样的类就是抽象类。 抽象类往往用来表征我们在对问题领域进行分析、 设…

【Git入门之五】版本管理

2019独角兽企业重金招聘Python工程师标准>>> 1.版本回退 我们先看一下从项目开始到现在做了什么操作。 [cpp] view plaincopy #总共是4个操作 $ git log --prettyoneline c5c83cfcdb25c67a5c66b4fe3844d0ea912830ec remove JackyData03 a25c58804cb3f4045564fc0e…

Java 进阶——单例模式

一、单例模式概念及特点 Java中单例模式是一种常见的设计模式,单例模式分三种:懒汉式单例、饿汉式单例、登记式单例三种。 单例模式有一下特点: 1、单例类只能有一个实例。 2、单例类必须自己自己创建自己的唯一实例。 3、单例类必须给所有其…

Java 关键字—— static 与 final

static表示“全局”或者“静态”的意思,用来修饰成员变量和成员方法,也可以形成静态static代码块,但是Java语言中没有全局变量的概念。 被static修饰的成员变量和成员方法独立于该类的任何对象。也就是说,它不依赖类特定的实例&am…

Java 三大特性之——继承

继承(inheritance)是面向对象的重要概念。继承是除组合(composition)之外,提高代码重复可用性(reusibility)的另一种重要方式。我们在组合(composition)中看到,组合是重复调用对象的功能接口。我们将看到,继承可以重复利用已有的类的定义。 类…

基于Linux的 Open×××网络之网络架构应用实例

基于Linux的 Open网络之网络架构应用实例Open 概述Open 是一个开源的加密隧道构建工具,基于 OpenSSL 的 SSL/TLS 协议,可以在 Internet中实现点对点的 SSL 安全连接。使用 Open 的好处是安全、易用和稳定,且认证方式灵活,具备实现…

Java 进阶——自动装箱和自动拆箱

1、什么是自动装箱拆箱 基本数据类型的自动装箱(autoboxing)、拆箱(unboxing)是自J2SE 5.0开始提供的功能。 一般我们要创建一个类的对象实例的时候,我们会这样: Class a new Class(parameter); 当我们创建一个Integer对象时,却可以这样&…

基于KVM的虚拟化研究及应用

引言 虚拟化技术是IBM在20世纪70年代首先应用在IBM/370大型机上,这项技术极大地提高了大型机资源利用率。随着软硬件技术的迅速发展,这项属于大型机及专利的技术开始在普通X86计算机上应用并成为当前计算机发展和研究的一个热点方向。目前&am…

Java 进阶—— super 和 this 的用法

一、this Java关键字this只能用于方法方法体内。当一个对象创建后,Java虚拟机(JVM)就会给这个对象分配一个引用自身的指针,这个指针的名字就是this。因此,this只能在类中的非静态方法中使用,静态方…

unity中脚本编辑器UnIDE

引言 unity默认脚本编辑器是MonoDevelop,随着unity4.3面世,MonoDevelop (4.0.1)版本也随之而来,更新为界面更改和bug自动修复功能等,具体还未使用。 点击unity的Edit下的属性(preference),可以更改默认脚本编辑器&…

apk,task,进程区别

2019独角兽企业重金招聘Python工程师标准>>> apk,task,进程区别 apk一般占一个dalvik,一个进程,一个task。通过设置也可以多个进程,占多个task。 task是一个activity的栈,其中"可能"含有来自…

Java 线程 —— 基础篇

一、操作系统中线程和进程的概念 现在的操作系统是多任务操作系统。多线程是实现多任务的一种方式。 进程是指一个内存中运行的应用程序,每个进程都有自己独立的一块内存空间,一个进程中可以启动多个线程。比如在Windows系统中,一个运行的exe…

【Java并发性和多线程】线程安全及不可变性

2019独角兽企业重金招聘Python工程师标准>>> 本文为转载学习 原文链接:http://tutorials.jenkov.com/java-concurrency/thread-safety-and-immutability.html 译文链接:http://ifeve.com/thread-safety-and-immutability/ 当多个线程同时访问…

Java 高级—— IO 基础

一、File 类 先看一下File 类的定义 [java] view plaincopy public class File extends Object implements Serizliable Comparable<File> 从定义看&#xff0c;File类是Object的直接子类&#xff0c;同时它继承了Comparable接口可以进行数组的排序。 File类的操作包括…

安装 SharePoint 2013 Foundation

一、Foundation版本的区别Foundation版本的区别见附件&#xff08;英文&#xff09;。官网下载地址 http://www.microsoft.com/zh-cn/download/details.aspx?id35488二、安装必备软件三、独立安装模式1. 启动安装向导2. 接受软件许可条款3. 选择服务器类型和数据位置4. 结束安…

Java 异常处理机制

异常处理是程序设计中一个非常重要的方面&#xff0c;也是程序设计的一大难点&#xff0c;从C开始&#xff0c;你也许已经知道如何用if...else...来控制异常了&#xff0c;也许是自发的&#xff0c;然而这种控制异常痛苦&#xff0c;同一个异常或者错误如果多个地方出现&#x…

架构师未来性的基础:简单性

作者&#xff1a;高焕堂&#xff0c;misoo.twqq.com 首页&#xff1a;Backee e架构师未来性的基础&#xff1a;简单性 Apple公司创始人乔布斯(Steve Jobs)曾说到&#xff1a;“简单比复杂更难&#xff0c;你必须努力让你的想法变得清晰…

Android 基础—— 对Context的理解与使用技巧

一、Context 基础概念 1、什么是Context 1) Context是一个抽象类&#xff0c;其通用实现在ContextImpl类中。 2) Context&#xff1a;是一个访问application环境全局信息的接口&#xff0c;通过它可以访问application的资源和相关的类&#xff0c;其主要功能如下&a…

Android 四大组件 —— 广播(广播机制解析)

在网络通信中&#xff0c;一个IP网络范围中最大的IP 地址是被保留作为广播地址来使用的。比如某个网络的IP 范围是192.168.0.XXX&#xff0c;子网掩码是255.255.255.0&#xff0c;那么这个网络的广播地址就是192.168.0.255。广播数据包会被发送到同一网络上的所有端口&#xff…

Android 基础 —— 活动的生存周期

一、返回栈 Android 中的活动是可以层叠的。我们每启动一个新的活动&#xff0c;就会覆盖在原活动之上&#xff0c;然后点击Back 键会销毁最上面的活动&#xff0c;下面的一个活动就会重新显示出来。 其实Android 是使用任务&#xff08;Task&#xff09;来管理活动的&#xff…