Java核心类库:内部类那点事儿

内部类:定义在类的内部的类

  为什么需要内部类?

  ● 典型的情况是,内部类继承自某个类或实现某个接口,内部类的代码操作创建其的外围类的对象。所以你可以认为内部类提供了某种进入其外围类的窗口。

  ● java中的内部类和接口加在一起,可以实现多继承。

  ● 可以使某些编码根简洁。

  ● 隐藏你不想让别人知道的操作。

  使用内部类最吸引人的原因是:

   每个内部类都能独立地继承自一个(接口的)实现,所以无论外围类是否已经继承了某个(接口的)实现,对于内部类都没有影响。如果没有内部类提供的可以继 承多个具体的或抽象的类的能力,一些设计与编程问题就很难解决。从这个角度看,内部类使得多重继承的解决方案变得完整。接口解决了部分问题,而内部类有效 地实现了“多重继承”。

  内部类分为: 成员内部类、静态嵌套类、方法内部类、匿名内部类。

  特点:

  一、内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.class文件,但是前面冠以外部类的类命和$符号。

  二、内部类可以直接或利用引用访问外部类的属性和方法,包括私有属性和方法(但静态内部类不能访问外部类的非静态成员变量和方法)。内部类所访问的外部属性的值由构造时的外部类对象决定。

  三、而外部类要访问内部类的成员,则只能通过引用的方式进行,可问内部类所有成员

  四、访问机制:

  1. System.out.println(this.x);或System.out.println(x);//内部类访问内部类的成员变量或成员方法可用此方法。 
  2. System.out.println(OuterClass.this.x);//内部类访问外部类的同名变量时可用此方法,如果没有同名可用System.out.println(x);

  五、内部类可以使用任意的范围限定:public/private/protected class InnerClass,且严格按照这几种访问权限来控制内部类能使用的范围。普通类的范围限定只可以是public或者不加。

  六、内部类的命名不允许与外部类 重名,内部类可以继承同级的内部类,也可继承其它类(除内部类和外部类)。

  七、内部类可以定义为接口,并且可以定义另外一个类来实现它

  八、内部类可以定义为抽象类,可以定义另外一个内部类继承它

  九、内部类使用static修饰,自动升级为顶级类,外部类不可以用static修饰,用OuterClass.InnerClass inner=new OuterClass.InnerClass();创建实例。内部类还可定义为final.

  十、内部类可以再定义内部类(基本不用)

  十一、方法内的内部类:

  方法内的内部类不能加范围限定(protected public private)

  方法内的内部类不能加static修饰符

  方法内的内部类只能在方法内构建其实例

  方法内的内部类如果访问方法局部变量,则此局部变量必须使用final修饰

  1)静态内部类(静态嵌套类)

  从技术上讲,静态嵌套类不属于内部类。因为内部类与外部类共享一种特殊关系,更确切地说是对实例的共享关系。而静态嵌套类则没有上述关系。它只是位置在另一个类的内部,因此也被称为顶级嵌套类。

  静态的含义是该内部类可以像其他静态成员一样,没有外部类对象时,也能够访问它。静态嵌套类不能访问外部类的成员和方法。

 语法

  1. package com.tarena.day13;  
  2.  
  3. import com.tarena.day13.Foo.Koo;  
  4. /**  
  5.  * 静态类内部语法演示  
  6.  */ 
  7. public class StaticInner {  
  8.  public static void main(String[] args) {  
  9.   Koo koo = new Koo();  
  10.   System.out.println(koo.add());//4  
  11.  }  
  12.  
  13. }  
  14. class Foo{  
  15.  int a = 1;  
  16.  static int b = 3;  
  17.  /** 静态内部类,作用域类似于静态变量,属于类的 */ 
  18.  static class Koo{  
  19.   public int add(){  
  20.    //a ,不能访问a  
  21.    return b+1;  
  22.   }  
  23.  }  
  24. }

  2)成员内部类

  * 1 成员内部类必须利用外部类实例创建

  * 2 成员内部类可以共享外部类的实例变量

  1. import com.tarena.day13.inn.Goo.Moo;  
  2.  
  3. public class InnerClassDemo {  
  4.  public static void main(String[] args) {  
  5.   //Moo moo = new Moo(); //编译错误,必须创建Goo的实例  
  6.   Goo goo = new Goo();  
  7.   Moo moo = goo.new Moo();//利用goo实例创建Moo实例  
  8.   Moo moo1 = goo.new Moo();  
  9.   //moo和moo1共享同一个goo实例的实例变量  
  10.   System.out.println(moo.add());//2  
  11.   System.out.println(moo1.add());//2  
  12.   Goo goo1 = new Goo();  
  13.   goo1.a = 8;  
  14.   Moo m1 = goo1.new Moo();  
  15.   Moo m2 = goo1.new Moo();  
  16.   System.out.println(m1.add());//9  
  17.   System.out.println(m2.add());//9  
  18.     
  19.  }  
  20. }  
  21. class Goo{  
  22.  int a = 1;  
  23.  /**成员内部类*/ 
  24.  class Moo{  
  25.   public int add(){  
  26.    return a+1;  
  27.   }  
  28.  }  
  29. }

  3)局部内部类(方法内部类)

  (1)方法内部类只能在定义该内部类的方法内实例化,不可以在此方法外对其实例化。

  (2)方法内部类对象不能使用该内部类所在方法的非final局部变量。

  因为方法的局部变量位于栈上,只存在于该方法的生命期内。当一个方法结束,其栈结构被删除,局部变量成为历史。但是该方法结束之后,在方法内创 建的内部类对象可能仍然存在于堆中!例如,如果对它的引用被传递到其他某些代码,并存储在一个成员变量内。正因为不能保证局部变量的存活期和方法内部类对 象的一样长,所以内部类对象不能使用它们。用法

  1. package com.tarena.day13.inn;  
  2.  
  3. import java.util.Comparator;  
  4.  
  5. /**  
  6.  * 局部内部类  
  7.  */ 
  8. public class LocalInnerClassDemo {  
  9.  public static void main(String[] args) {  
  10.   int a = 5;  
  11.   final int b = 5;  
  12.   //局部内部类,定义在方法内部,作用域类似于局部变量  
  13.   //仅仅在方法内部可见  
  14.   //在局部内部类中可以访问方法中的局部final变量  
  15.   class Foo{  
  16.    public int add(){  
  17.     return  b;//正确  
  18.     //return a;//编译错误  
  19.    }  
  20.   }  
  21.     
  22.   Foo foo = new Foo();  
  23.   //临时的自定义比较规则  
  24.   class ByLength implements Comparator<String>{  
  25.    public int compare(String o1,String o2){  
  26.     return o1.length()-o2.length();  
  27.    }  
  28.   }  
  29.  }  
  30.  
  31. }

  4)匿名内部类

  顾名思义,没有名字的内部类。表面上看起来它们似乎有名字,实际那不是它们的名字。

  匿名内部类就是没有名字的内部类。什么情况下需要使用匿名内部类?如果满足下面的一些条件,使用匿名内部类是比较合适的:

  只用到类的一个实例。

  ● 类在定义后马上用到。

  ● 类非常小(SUN推荐是在4行代码以下)

  ● 给类命名并不会导致你的代码更容易被理解

  在使用匿名内部类时,要记住以下几个原则:

  ● 匿名内部类不能有构造方法。

  ● 匿名内部类不能定义任何静态成员、方法和类。

  ● 匿名内部类不能是public,protected,private,static。

  ● 只能创建匿名内部类的一个实例。

  ● 一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类。

  ● 因匿名内部类为局部内部类,所以局部内部类的所有限制都对其生效。

  A、继承式的匿名内部类和接口式的匿名内部类。

  1. import java.util.Arrays;  
  2. import java.util.Comparator;  
  3.  
  4. /**匿名内部类 语法*/ 
  5. public class AnnInnerClass {  
  6.  
  7.  public static void main(String[] args) {  
  8.   // TODO Auto-generated method stub  
  9.   Yoo yoo = new Yoo();//创建Yoo的实例  
  10.   Yoo y1 = new Yoo(){};  
  11.   //new Yoo(){}创建匿名类实例  
  12.   //匿名类new Yoo(){}是继承Yoo类,并且同时创建了对象  
  13.   //new Yoo(){}是Yoo的子类型,其中{}是类体(class Body)  
  14.   //类体中可以定义任何类内的语法,如:属性,方法,方法重载,方法覆盖,等  
  15.   //子类型没有名字,所以叫匿名类!  
  16.   Yoo y2 = new Yoo(){  
  17.    public String toString(){//方法重写(覆盖)  
  18.     return "y2"; //y2是子类的实例  
  19.    }  
  20.   };  
  21.   System.out.println(y2);//"y2",调用了匿名类对象toString()  
  22.   //匿名内部类可以继承/实现 于 类,抽象类,接口等  
  23.   //按照继承的语法,子类型必须实现所有的抽象方法  
  24.     
  25.   //Xoo x = new Xoo(){};//编译错误,没有实现方法  
  26.   final int b = 5;  
  27.   Xoo xoo = new Xoo(){ //是实现接口,并且创建匿名类实例,不是创建接口对象  
  28.    public int add(int a){//实现接口中的抽象方法  
  29.     return a+b; //要访问局部变量b,只能访问final变量  
  30.    }  
  31.   };  
  32.   System.out.println(xoo.add(5));//10,调用对象的方法  
  33.   //Comparator接口也可以使用匿名类的方式  
  34.   Comparator<String> byLength = new Comparator<String>(){  
  35.    public int compare(String o1,String o2){  
  36.     return o1.length()-o2.length();  
  37.       
  38.    }  
  39.   };  
  40.   String[] names = {"Andy","Tom","Jerry"};  
  41.   Arrays.sort(names,byLength);  
  42.   System.out.println(Arrays.toString(names));  
  43.   //也可以这样写,工作中常用  
  44.   Arrays.sort(names,new Comparator<String>(){  
  45.    public int compare(String o1,String o2){  
  46.     return o1.length()-o2.length();  
  47.    }  
  48.   });  
  49.  }  
  50.  
  51. }

 接口式的匿名内部类是实现了一个接口的匿名类。而且只能实现一个接口。

  B. 参数式的匿名内部类。

  1. class Bar{   
  2.     void doStuff(Foo f){  
  3.      }   
  4. }   
  5. interface Foo{   
  6.     void foo();   
  7. }   
  8. class Test{    
  9.  static void go(){    
  10.      Bar b = new Bar();    
  11.      b.doStuff(new Foo(){    
  12.          public void foo(){    
  13.              System.out.println("foofy");    
  14.          }   
  15.       });   
  16.  }   
  17. }

  构造内部类对象的方法有:

  1、内部类在自己所处的外部类的静态方法内构建对象或在另一个类里构造对象时应用如下形式:

  (1)

  1. OuterClass out = new OuterClass();  
  2. OuterClass.InnerClass in = out.new InnerClass();

  (2)

OuterClass.InnerClass in=new OuterClass().new InnerClass();

  其中OuterClass是外部类,InnerClass是内部类。

  2、内部类在它所在的外部类的非静态方法里或定义为外部类的成员变量时,则可用以下方式来构造对象:

InnerClass in = new InnerClass();

  3、如果内部类为静态类,则可用如下形式来构造函数:

OuterClass.InnerClass in = new OuterClass.InnerClass();

  无需再利用外部类的对象来来构造内部类对象,如果静态内部类需要在静态方法或其它类中构造对象就必须用上面的方式来初始化。

本文出自seven的测试人生公众号最新内容请见作者的GitHub页:http://qaseven.github.io/

时间: 2024-12-03 21:46:21

Java核心类库:内部类那点事儿的相关文章

java-Java使用的核心类库是JRE,C++没有类似自己的核心类库,这就是Java可移植性的原因么

问题描述 Java使用的核心类库是JRE,C++没有类似自己的核心类库,这就是Java可移植性的原因么 我只接触过java,而且只用过windows.一直都知道java与其他语言区别之一是可移植性,但具体是什么一直说不清楚. 目前在看swing和awt等关于图形的,有说awt是重量级的依赖于操作系统绘图,移植其他平台,画风大变,效果很差:swing是轻量的,不依赖. 是不是awt底层调用了操作系统已有类库,导致移植其他平台效果很差,而swing自己用底层语言一笔一划做出图形,不用操作系统原有库,

Java核心编程卷1:基础

现在已经步入了21世纪,我们很难想象曾经Java相关的图书是那样的稀少,但这就是Java刚出现时的情况.那时所有与Java相关的图书基本上都是由Sun公司的几个工程师们所发布的,比如说Arthur Van Hoff et.al所写的"Hooked on Java"以及Patrick Naughton所写的"The Java Handbook". 当然了,这些图书填补了Java的空白,不过每本书都有3.400页,这些图书都假设读者了解程序,特别是面向对象程序设计. 1

Java核心编程实践--视频

Java核心编程实践--视频 下载地址:http://v.51work6.com/courseInfoRedirect.do?action=courseInfo&courseId=240577    图形用户界面的开发是本教程的重点,图书管理系统是使用图形界面开发的.集合框架是非常重要的内容,通过本教程的学习使学员掌握两大接口Collection和Map,还有List等,以及它们的实现类ArrayList.Vector和HashMap等.使用Collections类实现对集合的排序.最大值.最小

Guava v20.0-rc1,Google 的 Java 常用类库

Guava v20.0-rc1 发布了,Google 的 Java 常用类库.该版本主要更新内容是设置版本号为20.0 RC1. 详细改进内容请看:https://github.com/google/guava/compare/v20.0-rc1...master 下载地址: Source code (zip) Source code (tar.gz) Guava 中文是石榴的意思,该项目是 Google 的一个开源项目,包含许多 Google 核心的 Java 常用库. 目前主要包含: com

JAVA常用类库简介(转)

Java编程语言中为方便学习者学习,编制了许多类,这些类已经经过测试,都是我们编程的基础.如果不利用这些已存在的类,我们的编程工作将变得异常复杂并且效率低下.所以我们应尽可能多的掌握Java基本类库的内容.类库中的类按照其用途归属于不同的包中. 1.java.lang包 java.lang包 是Java中最常用的包,程序不需要注入,就可以使用该包中的类,利用包中的类可以设计最基本的Java程序. 2.java.awt包 java.awt包中的类提供了图形界面的创建方法,包括按钮.文本框.列表框.

请问java中类库跟包的区别

问题描述 请问java中类库跟包的区别 请问类库跟包的区别是什么? 解决方案 类库包含很多个包 解决方案二: 库通常是一个组功能的集合,像Spring就是一个类库,可以理解为一个工程:而包是类的容器,用于分隔类名空间,是一个工程下的一部分. 解决方案三: 你只需要知道类库包含包就可以了 解决方案四: 类库字面意思就是类的集合,类库不等于框架,Framework 是比类库更高一级的概念.类库是一个综合性的面向对象的可重用类型集合,这些类型包括:接口.抽象类和具体类.类库可以单独从第三方软件商获得.

java内部类-java实验题 内部类的定义与使用

问题描述 java实验题 内部类的定义与使用 定义一个Cube类,其中包含一个变量height,一个方法:计算立方体的体积(带形参,形参为面积).并在其内部定义一个Rectangle类,其中包含两个变量width和height,一个方法:计算底面积. 解决方案 class Cube { private double height; Cube() {} Cube(double height) { this.height = height; } public double volume(double

Java 中的内部类和匿名类

Java 内部类有什么好处?为什么需要内部类? 首先举一个简单的例子,如果你想实现一个接口,但是这个接口中的一个方法和你构想的这个类中的一个方法的名称,参数相同,你应该怎么办?这时候,你可以建一个内部类实现这个接口.由于内部类对外部类的所有内容都是可访问的,所以这样做可以完成所有你直接实现这个接口的功能. 不过你可能要质疑,更改一下方法的不就行了吗? 的确,以此作为设计内部类的理由,实在没有说服力. 真正的原因是这样的,java 中的内部类和接口加在一起,可以的解决常被 C++ 程序员抱怨 ja

源码-java核心内裤java.awt.color如何通过value变量实现功能的,请教下各位。

问题描述 java核心内裤java.awt.color如何通过value变量实现功能的,请教下各位. 想研究java核心代码的实现问题,遇到以下问题. 源码中初始化代码: public Color(int r int g int b int a) { value = ((a & 0xFF) << 24) | ((r & 0xFF) << 16) | ((g & 0xFF) << 8) | ((b & 0xFF) << 0);