您当前的位置:首页 > 互联网教程

java内部类与匿名内部类作用分别是什么

发布时间:2025-05-21 06:46:28    发布人:远客网络

java内部类与匿名内部类作用分别是什么

一、java内部类与匿名内部类作用分别是什么

Java内部类 \x0d\x0a分四种:成员内部类、局部内部类、静态内部类和匿名内部类。 \x0d\x0a1、成员内部类:即作为外部类的一个成员存在,与外部类的属性、方法并列。\x0d\x0a注意:成员内部类中不能定义静态变量,但可以访问外部类的所有成员。\x0d\x0apublic class Outer{\x0d\x0aprivate static int i= 1;\x0d\x0aprivate int j=10;\x0d\x0aprivate int k=20;\x0d\x0apublic static void outer_f1(){\x0d\x0a//do more something\x0d\x0a}\x0d\x0apublic void out_f2(){\x0d\x0a//do more something\x0d\x0a}\x0d\x0a\x0d\x0a//成员内部类\x0d\x0aclass Inner{\x0d\x0a//static int inner_i=100;//内部类中不允许定义静态变量\x0d\x0aint j=100;//内部类中外部类的实例变量可以共存\x0d\x0aint inner_i=1;\x0d\x0avoid inner_f1(){\x0d\x0a System.out.println(i);//外部类的变量如果和内部类的变量没有同名的,则可以直接用变量名访问外部类的变量\x0d\x0a System.out.println(j);//在内部类中访问内部类自己的变量直接用变量名\x0d\x0a System.out.println(this.j);//也可以在内部类中用"this.变量名"来访问内部类变量\x0d\x0a//访问外部类中与内部类同名的实例变量可用"外部类名.this.变量名"。\x0d\x0a System.out.println(k);//外部类的变量如果和内部类的变量没有同名的,则可以直接用变量名访问外部类的变量\x0d\x0a outer_f1();\x0d\x0a outer_f2();\x0d\x0a}\x0d\x0a}\x0d\x0a//外部类的非静态方法访问成员内部类\x0d\x0apublic void outer_f3(){\x0d\x0a Inner inner= new Inner();\x0d\x0a inner.inner_f1();\x0d\x0a}\x0d\x0a\x0d\x0a//外部类的静态方法访问成员内部类,与在外部类外部访问成员内部类一样\x0d\x0apublic static void outer_f4(){\x0d\x0a//step1建立外部类对象\x0d\x0a Outer out= new Outer();\x0d\x0a//***step2根据外部类对象建立内部类对象***\x0d\x0a Inner inner=out.new Inner();\x0d\x0a//step3访问内部类的方法\x0d\x0a inner.inner_f1();\x0d\x0a}\x0d\x0a\x0d\x0apublic static void main(String[] args){\x0d\x0a outer_f4();\x0d\x0a}\x0d\x0a}\x0d\x0a成员内部类的优点:\x0d\x0a⑴内部类作为外部类的成员,可以访问外部类的私有成员或属性。(即使将外部类声明为PRIVATE,但是对于处于其内部的内部类还是可见的。)\x0d\x0a⑵用内部类定义在外部类中不可访问的属性。这样就在外部类中实现了比外部类的private还要小的访问权限。\x0d\x0a注意:内部类是一个编译时的概念,一旦编译成功,就会成为完全不同的两类。对于一个名为outer的外部类和其内部定义的名为inner的内部类。编译完成后出现outer.class和outer$inner.class两类。\x0d\x0a2、局部内部类:即在方法中定义的内部类,与局部变量类似,在局部内部类前不加修饰符public或private,其范围为定义它的代码块。\x0d\x0a注意:局部内部类中不可定义静态变量,可以访问外部类的局部变量(即方法内的变量),但是变量必须是final的。\x0d\x0apublic class Outer{\x0d\x0a private int s= 100;\x0d\x0a private int out_i= 1;\x0d\x0a public void f(final int k){\x0d\x0a final int s= 200;\x0d\x0a int i= 1;\x0d\x0a final int j= 10;\x0d\x0a class Inner{//定义在方法内部\x0d\x0a int s= 300;//可以定义与外部类同名的变量\x0d\x0a//static int m= 20;//不可以定义静态变量\x0d\x0a Inner(int k){\x0d\x0a inner_f(k);\x0d\x0a}\x0d\x0a int inner_i= 100;\x0d\x0a void inner_f(int k){\x0d\x0a System.out.println(out_i);//如果内部类没有与外部类同名的变量,在内部类中可以直接访问外部类的实例变量\x0d\x0a System.out.println(k);//*****可以访问外部类的局部变量(即方法内的变量),但是变量必须是final的*****\x0d\x0a// System.out.println(i);\x0d\x0a System.out.println(s);//如果内部类中有与外部类同名的变量,直接用变量名访问的是内部类的变量\x0d\x0a System.out.println(this.s);//用"this.变量名"访问的也是内部类变量\x0d\x0a System.out.println(Outer.this.s);//用外部"外部类类名.this.变量名"访问的是外部类变量\x0d\x0a}\x0d\x0a}\x0d\x0a new Inner(k);\x0d\x0a}\x0d\x0a\x0d\x0a public static void main(String[] args){\x0d\x0a//访问局部内部类必须先有外部类对象\x0d\x0a Outer out= new Outer();\x0d\x0a out.f(3);\x0d\x0a}\x0d\x0a\x0d\x0a}\x0d\x0a\x0d\x0a注意:\x0d\x0a在类外不可直接生成局部内部类(保证局部内部类对外是不可见的)。要想使用局部内部类时需要生成对象,对象调用方法,在方法中才能调用其局部内部类。通过内部类和接口达到一个强制的弱耦合,用局部内部类来实现接口,并在方法中返回接口类型,使局部内部类不可见,屏蔽实现类的可见性。\x0d\x0a\x0d\x0a3、静态内部类:静态内部类定义在类中,任何方法外,用static定义。\x0d\x0a注意:静态内部类中可以定义静态或者非静态的成员\x0d\x0apublic class Outer{\x0d\x0a private static int i= 1;\x0d\x0a private int j= 10;\x0d\x0a public static void outer_f1(){\x0d\x0a \x0d\x0a}\x0d\x0a public void outer_f2(){\x0d\x0a \x0d\x0a}\x0d\x0a//静态内部类可以用public,protected,private修饰\x0d\x0a//静态内部类中可以定义静态或者非静态的成员\x0d\x0a static class Inner{\x0d\x0a static int inner_i= 100;\x0d\x0a int inner_j= 200;\x0d\x0a static void inner_f1(){\x0d\x0a System.out.println("Outer.i"+i);//静态内部类只能访问外部类的静态成员\x0d\x0a outer_f1();//包括静态变量和静态方法\x0d\x0a}\x0d\x0a void inner_f2(){\x0d\x0a// System.out.println("Outer.i"+j);//静态内部类不能访问外部类的非静态成员\x0d\x0a// outer_f2();//包括非静态变量和非静态方法\x0d\x0a} \x0d\x0a \x0d\x0a}\x0d\x0a \x0d\x0a public void outer_f3(){\x0d\x0a//外部类访问内部类的静态成员:内部类.静态成员\x0d\x0a System.out.println(Inner.inner_i);\x0d\x0a Inner.inner_f1();\x0d\x0a//外部类访问内部类的非静态成员:实例化内部类即可\x0d\x0a Inner inner= new Inner();\x0d\x0a inner.inner_f2();\x0d\x0a \x0d\x0a}\x0d\x0a public static void main(String[] args){\x0d\x0a new Outer().outer_f3();\x0d\x0a}\x0d\x0a\x0d\x0a}\x0d\x0a\x0d\x0a注意:*******生成(new)一个静态内部类不需要外部类成员:这是静态内部类和成员内部类的区别。静态内部类的对象可以直接生成:\x0d\x0aOuter.Inner in=new Outer.Inner();\x0d\x0a而不需要通过生成外部类对象来生成。这样实际上使静态内部类成为了一个顶级类。静态内部类不可用private来进行定义。*******\x0d\x0a\x0d\x0a例子:\x0d\x0a对于两个类,拥有相同的方法:\x0d\x0aclass People\x0d\x0a{\x0d\x0a run();\x0d\x0a}\x0d\x0aclass Machine{\x0d\x0a run();\x0d\x0a}\x0d\x0a此时有一个robot类:\x0d\x0aclass Robot extends People implement Machine.\x0d\x0a此时run()不可直接实现。\x0d\x0a注意:当类与接口(或者是接口与接口)发生方法命名冲突的时候,此时必须使用内部类来实现。用接口不能完全地实现多继承,用接口配合内部类才能实现真正的多继承。\x0d\x0a\x0d\x0a4、匿名内部类 \x0d\x0a匿名内部类是一种特殊的局部内部类,它是通过匿名类实现接口。\x0d\x0aIA被定义为接口。\x0d\x0aIA I=new IA(){};\x0d\x0a\x0d\x0a匿名内部类的特点:\x0d\x0a\x0d\x0a1,一个类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的事先或是覆盖。\x0d\x0a2,只是为了获得一个对象实例,不需要知道其实际类型。\x0d\x0a3,类名没有意义,也就是不需要使用到。\x0d\x0a\x0d\x0apublic class Outer{\x0d\x0a private static int i= 1;\x0d\x0a private int j= 10;\x0d\x0a public static void outer_f1(){\x0d\x0a \x0d\x0a}\x0d\x0a public void outer_f2(){\x0d\x0a \x0d\x0a}\x0d\x0a//静态内部类可以用public,protected,private修饰\x0d\x0a//静态内部类中可以定义静态或者非静态的成员\x0d\x0a static class Inner{\x0d\x0a static int inner_i= 100;\x0d\x0a int inner_j= 200;\x0d\x0a static void inner_f1(){\x0d\x0a System.out.println("Outer.i"+i);//静态内部类只能访问外部类的静态成员\x0d\x0a outer_f1();//包括静态变量和静态方法\x0d\x0a}\x0d\x0a void inner_f2(){\x0d\x0a// System.out.println("Outer.i"+j);//静态内部类不能访问外部类的非静态成员\x0d\x0a// outer_f2();//包括非静态变量和非静态方法\x0d\x0a}\x0d\x0a}\x0d\x0a \x0d\x0a public void outer_f3(){\x0d\x0a//外部类访问内部类的静态成员:内部类.静态成员\x0d\x0a System.out.println(Inner.inner_i);\x0d\x0a Inner.inner_f1();\x0d\x0a//外部类访问内部类的非静态成员:实例化内部类即可\x0d\x0a Inner inner= new Inner();\x0d\x0a inner.inner_f2();\x0d\x0a \x0d\x0a}\x0d\x0a public static void main(String[] args){\x0d\x0a new Outer().outer_f3();\x0d\x0a}\x0d\x0a\x0d\x0a}\x0d\x0a\x0d\x0a注:一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类,没有类名,根据多态,我们使用其父类名。因他是局部内部类,那么局部内部类的所有限制都对其生效。匿名内部类是唯一一种无构造方法类。大部分匿名内部类是用于接口回调用的。匿名内部类在编译的时候由系统自动起名Out$1.class。如果一个对象编译时的类型是接口,那么其运行的类型为实现这个接口的类。因匿名内部类无构造方法,所以其使用范围非常的有限。当需要多个对象时使用局部内部类,因此局部内部类的应用相对比较多。匿名内部类中不能定义构造方法。如果一个对象编译时的类型是接口,那么其运行的类型为实现这个接口的类。\x0d\x0a\x0d\x0a________________________________________________________________________________\x0d\x0a\x0d\x0a内部类总结:\x0d\x0a1.首先,把内部类作为外部类的一个特殊的成员来看待,因此它有类成员的封闭等级:private,protected,默认(friendly),public\x0d\x0a它有类成员的修饰符: static,final,abstract\x0d\x0a2.非静态内部类nested inner class,内部类隐含有一个外部类的指针this,因此,它可以访问外部类的一切资源(当然包括private)\x0d\x0a外部类访问内部类的成员,先要取得内部类的对象,并且取决于内部类成员的封装等级。\x0d\x0a非静态内部类不能包含任何static成员.\x0d\x0a3.静态内部类:static inner class,不再包含外部类的this指针,并且在外部类装载时初始化.\x0d\x0a静态内部类能包含static或非static成员.\x0d\x0a静态内部类只能访问外部类static成员.\x0d\x0a外部类访问静态内部类的成员,循一般类法规。对于static成员,用类名.成员即可访问,对于非static成员,只能\x0d\x0a用对象.成员进行访问\x0d\x0a\x0d\x0a4.对于方法中的内部类或块中内部类只能访问块中或方法中的final变量。\x0d\x0a\x0d\x0a类成员有两种static, non-static,同样内部类也有这两种\x0d\x0anon-static内部类的实例,必须在外部类的方法中创建或通过外部类的实例来创建(OuterClassInstanceName.new innerClassName(ConstructorParameter)),并且可直接访问外部类的信息,外部类对象可通过OuterClassName.this来引用\x0d\x0astatic内部类的实例,直接创建即可,没有对外部类实例的引用。\x0d\x0a内部类不管static还是non-static都有对外部类的引用\x0d\x0anon-static内部类不允许有static成员\x0d\x0a\x0d\x0a方法中的内部类只允许访问方法中的final局部变量和方法的final参数列表,所以说方法中的内部类和内部类没什麽区别。但方法中的内部类不能在方法以外访问,方法中不可以有static内部类\x0d\x0a匿名内部类如果继承自接口,必须实现指定接口的方法,且无参数 \x0d\x0a匿名内部类如果继承自类,参数必须按父类的构造函数的参数传递

二、哪位朋友能帮我详解一下Java的内部类和匿名内部类

一、定义:在一个类内部定义的类。

二、类型:成员内部类、局部内部类(也叫方法内部类)、匿名内部类、静态嵌套类。

i.内部类仍然是一个独立的类,在编译后内部类会被编译成独立的.class文件,只是在前面加上外部类的类名和$符号。

ii.内部类不能用普通的方式访问。内部类是外部类的一个成员,因此内部类可以自由访问外部类的成员变量,无论是否是private的。

iii.内部类的this引用:内部类中同样使用this.成员变量,如要使用外部类的成员变量,则使用外部类名.this.成员变量。

i.成员内部类:它相当于外部类的一个成员变量或方法,可用修饰符为final、abstract、public、private、protected、strictfp和static。形式如下

编译上述代码会产生两个文件,Outer.class和Outer$Inner.class。成员内部类内不允许有任何静态声明!能够访问成员内部类的途径是通过外部类的对象或外部类中非静态的方法。

System.out.println(“Inner class”);

pulblic static void main(String args[]){

Outer.Inner in= out.new Inner();

ii.局部内部类(方法内部类):在外部类的某个方法内定义的类,与成员内部类不同,它相当于外部类的一个局部变量,修饰符只有final和abstract。

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

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

3.静态方法内的局部内部类没有this的引用。

iii.匿名内部类:没有名字的内部类。

System.out.println(“Driving a car!”);

public static void main(String[] args){

System.out.println(“Driving another car!”);

public static void main(String[] args){

3.参数匿名内部类:顾名思义,将匿名内部类当成参数使用。

iv.静态嵌套类:从技术上讲,静态嵌套类不属于内部类。因为内部类与外部类共享一种特殊关系,更确切地说是对实例的共享关系。而静态嵌套类则没有上述关系。它只是位置在另一个类的内部。因此也被称为顶级嵌套类。静态的含义是该内部类可以像其他静态成员一样,没有外部类对象时,也能够访问它。静态嵌套类不能访问外部类的成员和方法。

public static void main(String[] args){

Outer.Inner n= new Outer.Inner();

三、java内部类,匿名内部类这些是什么求详细用法,还有语法。

内部类是定义在另一个类中的类,使用它的原因主要有3个:

内部类方法可以访问该类定义所在的作用域中的数据,包括私有的数据;

内部类可以对同一个包中的其他类隐藏以来;

当想要定义一个回调函数且不想编写大量代码时,使用匿名内部类比较便捷。

【举例:《Java核心技术(卷I)》,6.4节——内部类,程序清单6-6】

*Thisprogramdemonstratesanonymousinnerclasses.

publicstaticvoidmain(String[]args)

TalkingClockclock=newTalkingClock(1000,true);

//keepprogramrunninguntiluserselects"Ok"

JOptionPane.showMessageDialog(null,"Quitprogram?");

*Aclockthatprintsthetimeinregularintervals.

*@paramintervaltheintervalbetweenmessages(inmilliseconds)

*@parambeeptrueiftheclockshouldbeep

publicTalkingClock(intinterval,booleanbeep)

ActionListenerlistener=newTimePrinter();

Timert=newTimer(interval,listener);

publicclassTimePrinterimplementsActionListener

publicvoidactionPerformed(ActionEventevent)

System.out.println("Atthetone,thetimeis"+now);

if(beep)Toolkit.getDefaultToolkit().beep();

}

上述代码中,TimePrinter就是在TalkingClock类的内部定义的一个内部类,因此它可以访问外围类的数据域,包括interval和beep这种私有域。

上述代码中,TimePrinter就是在TalkingClock类的内部定义的一个内部类,因此它可以访问外围类的数据域,包括interval和beep这种私有域。

讨论匿名内部类之前,先看看局部内部类,它是在一个方法中定义的类。代码示例如下,这里将上例中的TimePrinter类放到了start()方法中进行定义:

publicvoidstart(intinterval,finalbooleanbeep)

classTimePrinterimplementsActionListener

publicvoidactionPerformed(ActionEventevent)

System.out.println("Atthetone,thetimeis"+now);

if(beep)Toolkit.getDefaultToolkit().beep();

ActionListenerlistener=newTimePrinter();

Timert=newTimer(interval,listener);

}

局部内部类不能用public或private访问说明符来声明,它的作用域被限定在声明这个局部内部类的块中。局部内部类可以对外部世界完全地隐藏起来。

局部内部类不能用public或private访问说明符来声明,它的作用域被限定在声明这个局部内部类的块中。局部内部类可以对外部世界完全地隐藏起来。

局部内部类不仅能访问其外围类,还可以访问局部变量,不过这些局部变量必须被声明为final,如上述代码中start()方法的参数beep所示。

将上面的局部内部类的代码修改一下,就可以定义一个匿名内部类,这种类没有类名。

publicvoidstart(intinterval,finalbooleanbeep)

ActionListenerlistener=newActionListener()

publicvoidactionPerformed(ActionEventevent)

System.out.println("Atthetone,thetimeis"+now);

if(beep)Toolkit.getDefaultToolkit().beep();

Timert=newTimer(interval,listener);

}

请参照局部内部类的代码,比较不同之处。TimePrinter这个类名被省略了,定义listener这个局部内部类时,在其后的new ActionListener()后面跟了一个大括号括起的语句块,也就是此匿名内部类的定义语句。匿名内部类除了具有内部类的优点外,还可以减少代码量。

请参照局部内部类的代码,比较不同之处。TimePrinter这个类名被省略了,定义listener这个局部内部类时,在其后的new ActionListener()后面跟了一个大括号括起的语句块,也就是此匿名内部类的定义语句。匿名内部类除了具有内部类的优点外,还可以减少代码量。

【内容有些多,但愿能帮到你^_^】



相关内容FAQs: