Java嵌套類和內部類詳解_java

來源:互聯網
上載者:User

一、什麼是嵌套類及內部類?

    可以在一個類的內部定義另一個類,這種類稱為嵌套類(nested classes),它有兩種類型:
靜態嵌套類和非靜態嵌套類。靜態嵌套類使用很少,最重要的是非靜態嵌套類,也即是被稱作為
內部類(inner)。嵌套類從JDK1.1開始引入。其中inner類又可分為三種:

    其一、在一個類(外部類)中直接定義的內部類;
    其二、在一個方法(外部類的方法)中定義的內部類;
    其三、匿名內部類。

下面,我將說明這幾種嵌套類的使用及注意事項。

二、靜態嵌套類

    如下所示代碼為定義一個靜態嵌套類,

複製代碼 代碼如下:

public class StaticTest {
private static String name = "javaJohn";
private String id = "X001";

static class Person{
private String address = "swjtu,chenDu,China";
public String mail = "josserchai@yahoo.com";//內部類公有成員
public void display(){
//System.out.println(id);//不能直接存取外部類的非靜態成員
System.out.println(name);//只能直接存取外部類的靜態成員
System.out.println("Inner "+address);//訪問本內部類成員。
}
}

public void printInfo(){
Person person = new Person();
person.display();

//System.out.println(mail);//不可訪問
//System.out.println(address);//不可訪問

System.out.println(person.address);//可以訪問內部類的私人成員
System.out.println(person.mail);//可以訪問內部類的公有成員

}
public static void main(String[] args) {
StaticTest staticTest = new StaticTest();
staticTest.printInfo();
}
}

在靜態嵌套類內部,不能訪問外部類的非靜態成員,這是由Java文法中"靜態方法不能直接存取非靜態成員"所限定。
若想訪問外部類的變數,必須通過其它方法解決,由於這個原因,靜態嵌套類使用很少。注意,外部類訪問內
部類的的成員有些特別,不能直接存取,但可以通過內部類來訪問,這是因為靜態嵌套內的所有成員和方法預設為
靜態了。同時注意,內部靜態類Person只在類StaticTest 範圍內可見,若在其它類中引用或初始化,均是錯誤的。

三、在外部類中定義內部類

    如下所示代碼為在外部類中定義兩個內部類及它們的調用關係:

複製代碼 代碼如下:

public class Outer{
int outer_x = 100;

class Inner{
public int y = 10;
private int z = 9;
int m = 5;
public void display(){
System.out.println("display outer_x:"+ outer_x);
}
private void display2(){
System.out.println("display outer_x:"+ outer_x);
}

}

void test(){
Inner inner = new Inner();
inner.display();
inner.display2();
//System.out.println("Inner y:" + y);//不能訪問內部內變數
System.out.println("Inner y:" + inner.y);//可以訪問
System.out.println("Inner z:" + inner.z);//可以訪問
System.out.println("Inner m:" + inner.m);//可以訪問

InnerTwo innerTwo = new InnerTwo();
innerTwo.show();
}

class InnerTwo{
Inner innerx = new Inner();
public void show(){
//System.out.println(y);//不可訪問Innter的y成員
//System.out.println(Inner.y);//不可直接存取Inner的任何成員和方法
innerx.display();//可以訪問
innerx.display2();//可以訪問
System.out.println(innerx.y);//可以訪問
System.out.println(innerx.z);//可以訪問
System.out.println(innerx.m);//可以訪問
}
}

public static void main(String args[]){
Outer outer = new Outer();
outer.test();
}
}

以上代碼需要說明有,對於內部類,通常在定義類的class關鍵字前不加public 或 private等限制符,若加了
沒有任何影響,同時好像這些限定符對內部類的變數和方法也沒有影響(?)。另外,就是要注意,內部類Inner及
InnterTwo只在類Outer的範圍內是可知的,如果類Outer外的任何代碼嘗試初始化類Inner或使用它,編譯就不
會通過。同時,內部類的變數成員只在內部內內部可見,若外部類或同層次的內部類需要訪問,需採用樣本程式
中的方法,不可直接存取內部類的變數。

四、在方法中定義內部類
    如下所示代碼為在方法內部定義一個內部類:

public class FunOuter {
int out_x = 100;

public void test(){
class Inner{
String x = "x";
void display(){
System.out.println(out_x);
}
}
Inner inner = new Inner();
inner.display();
}

public void showStr(String str){
//public String str1 = "test Inner";//不可定義,只允許final修飾
//static String str4 = "static Str";//不可定義,只允許final修飾
String str2 = "test Inner";
final String str3 = "final Str";
class InnerTwo{
public void testPrint(){
System.out.println(out_x);//可直接存取外部類的變數
//System.out.println(str);//不可訪問本方法內部的非final變數
//System.out.println(str2);//不可訪問本方法內部的非final變數
System.out.println(str3);//只可訪問本方法的final型變數成員
}
}
InnerTwo innerTwo = new InnerTwo();
innerTwo.testPrint();
}

public void use(){
//Inner innerObj = new Inner();//此時Inner己不可見了。
//System.out.println(Inner.x);//此時Inner己不可見了。
}


public static void main(String[] args) {
FunOuter outer = new FunOuter();
outer.test();
}
}


從上面的常式我們可以看出定義在方法內部的內部類的可見度更小,它只在方法內部
可見,在外部類(及外部類的其它方法中)中都不可見了。同時,它有一個特點,就是方法
內的內部類連本方法的成員變數都不可訪問,它只能訪問本方法的final型成員。同時另一個
需引起注意的是方法內部定義成員,只允許final修飾或不加修飾符,其它像static等均不可用。

五、匿名內部類
    如下所示代碼為定義一個匿名內部類:匿名內部類通常用在Java的事件處理上

複製代碼 代碼如下:

import java.applet.*;
import java.awt.event.*;

    public class AnonymousInnerClassDemo extends Applet{
    public void init(){
        addMouseListener(new MouseAdapter(){
            public void mousePressed(MouseEvent me){
             showStatus("Mouse Pressed!");
        }
        })
    }
    public void showStatus(String str){
        System.out.println(str);
    }
    }

在上面的例子中,方法addMouseListener接受一個對象型的參數運算式,於是,在參數裡,我們定義了一個匿名內部類這個類是一個MouseAdapter類型的類,同時在這個類中定義了一個繼承的方法mousePressed,整個類做為一個參數。這個類沒有名稱,但是當執行這個運算式時它被自動執行個體化。同時因為,這個匿名內部類是定義在AnonymousInnerClassDemo 類內部的,所以它可以訪問它的方法showStatus。這同前面的內部類是一致的。

六、內部類使用的其它的問題

通過以上,我們可以清楚地看出內部類的一些使用方法,同時,在許多時候,內部類是在如Java的事件處理、或做為值對象來使用的。同時,我們需注意最後一個問題,那就是,內部類同其它類一樣被定義,同樣它也可以繼承外部其它包的類和實現外部其它地方的介面。同樣它也可以繼承同一層次的其它的內部類,甚至可以繼承外部類本身。下面我們給出最後一個例子做為結束:

複製代碼 代碼如下:

public class Layer {
//Layer類的成員變數
private String testStr = "testStr";

//Person類,基類
class Person{
String name;
Email email;
public void setName(String nameStr){
this.name = nameStr;
}
public String getName(){
return this.name;
}
public void setEmail(Email emailObj){

this.email = emailObj;
}
public String getEmail(){
return this.email.getMailStr();
}
//內部類的內部類,多層內部類
class Email{
String mailID;
String mailNetAddress;
Email(String mailId,String mailNetAddress){
this.mailID = mailId;
this.mailNetAddress = mailNetAddress;
}
String getMailStr(){
return this.mailID +"@"+this.mailNetAddress;
}
}
}
//另一個內部類繼承外部類本身
class ChildLayer extends Layer{
void print(){
System.out.println(super.testStr);//訪問父類的成員變數
}
}
//另個內部類繼承內部類Person
class OfficePerson extends Person{
void show(){
System.out.println(name);
System.out.println(getEmail());
}
}
//外部類的測試方法
public void testFunction(){
//測試第一個內部類
ChildLayer childLayer = new ChildLayer();
childLayer.print();

//測試第二個內部類
OfficePerson officePerson = new OfficePerson();
officePerson.setName("abner chai");
//注意此處,必須用 對象.new 出來對象的子類對象
//而不是Person.new Email(...)
//也不是new Person.Email(...)
officePerson.setEmail(officePerson.new Email("josserchai","yahoo.com"));

officePerson.show();
}
public static void main(String[] args) {
Layer layer = new Layer();
layer.testFunction();
}
}

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.