7.11. 内部类 (注:所有使用内部类的地方都可以不用内部类,但使用内部类可以使程序更加的简洁,便于命名规范和划分层次结构)。 内部类是指在一个外部类的内部再定义一个类。 *内部类可为静态,可用PROTECTED和PRIVATE修饰。(而外部类不可以:顶级类只能使用PUBLIC和DEFAULT)。 *JAVA文件中没有publie class 可以类名和文件不同名。 7.11.1. 内部类的分类 成员内部类、 局部内部类、 静态内部类、 匿名内部类(图形是要用到,必须掌握)。7.11.2. 成员内部类 作为外部类的一个成员存在,与外部类的属性、方法并列。 内部类和外部类的实例变量可以共存。 在内部类中访问实例变量:this.属性 在内部类访问外部类的实例变量:外部类名.this.属性。 在外部类的外部访问内部类,使用out.inner.成员内部类的特点: 1.内部类作为外部类的成员,可以访问外部类的私有成员或属性。(即使将外部类声明为PRIVATE,但是对于处于其内部的内部类还是可见的。) 2.用内部类定义在外部类中不可访问的属性。这样就在外部类中实现了比外部类的private还要小的访问权限。 注意:内部类是一个编译时的概念,一旦编译成功,就会成为完全不同的两类。 对于一个名为outer的外部类和其内部定义的名为inner的内部类。编译完成后出现outer.class和outer$inner.class两类。 3.成员内部类不能有静态属性建立内部类对象时应注意: 在外部类的内部可以直接使用inner s=new inner();(因为外部类知道inner是哪个类,所以可以生成对象。) 而在外部类的外部,要生成(new)一个内部类对象,需要首先建立一个外部类对象(外部类可用),然后在生成一个内部类对象。 Outer o=new Outer(); Outer.Inner in=o.new.Inner()。 7.11.3. 局部内部类 在方法中定义的内部类称为局部内部类。 与局部变量类似,在局部内部类前不加修饰符public和private,其范围为定义它的代码块。注意: 局部内部类不仅可以访问外部类实例变量,但可以访问外部类的局部常量 在类外不可直接访问局部内部类(保证局部内部类对外是不可见的)。 在方法中才能调用其局部内部类。7.11.4. 静态内部类 (注意:前三种内部类与变量类似,所以可以对照参考变量) 静态内部类定义在类中,任何方法外,用static定义。 静态内部类只能访问外部类的静态成员。 生成(new)一个静态内部类不需要外部类成员:这是静态内部类和成员内部类的区别。静态内部类的对象可以直接生成: Outer.Inner in=new Outer.Inner(); 而不需要通过生成外部类对象来生成。这样实际上使静态内部类成为了一个顶级类。静态内部类不可用private来进行定义。注意:当类与接口(或者是接口与接口)发生方法命名冲突的时候,此时必须使用内部类来实现。 用接口不能完全地实现多继承,用接口配合内部类才能实现真正的多继承。 例子: 对于两个类,拥有相同的方法: class People { run(); } interface Machine{ run(); } 此时有一个robot类: class Robot extends People implement Machine. 此时run()不可直接实现。 interface Machine { void run(); } class Person { void run(){System.out.println("run");} } class Robot extends Person { private class MachineHeart implements Machine { public void run(){System.out.println("heart run");} } public void run(){System.out.println("Robot run");} Machine getMachine(){return new MachineHeart();} } class Test { public static void main(String[] args) { Robot robot=new Robot(); Machine m=robot.getMachine(); m.run(); robot.run(); } } 7.11.5. 匿名内部类 匿名内部类是一种特殊的局部内部类,它是通过匿名类实现接口。 IA被定义为接口。 IA I=new IA(){}; 注:一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类,没有类名,根据多态,我们使用其父类名。 因其为局部内部类,那么局部内部类的所有限制都对其生效。 匿名内部类是唯一一种无构造方法类。 匿名内部类在编译的时候由系统自动起名Out$1.class。 如果一个对象编译时的类型是接口,那么其运行的类型为实现这个接口的类。 因匿名内部类无构造方法,所以其使用范围非常的有限。
另一种是有名字的:1.成员内部类 //成员内部类......相当于非静态方法 class MemberInner { private int a = 3; public class Inner2 { private int a = 2; public void doSomething() { // 调用外部类的属性 System.out.println(MemberInner.this.a);// 这块要注意......很重要!!! System.out.println(a); } } }public class Test3 { public static void main(String[] args) { MemberInner.Inner2 inner2 = new MemberInner().new Inner2();// 非静态内部类要new实例 inner2.doSomething(); } } 2.静态内部类//静态内部类......相当于静态方法 class StaticInner{ private static int a=3; public static class Inner{ public void test(){ System.out.println(a); } } }public class Test2 { public static void main(String[] args) { StaticInner.Inner inner=new StaticInner.Inner();//静态内部类直接调用 inner.test(); } }3.局部内部类 //局部内部类......相当于局部变量 class LocalInner { public void doSomething(int b) { final int a = 3;// 只能访问final的变量 class Inner3 { public void test(int b) { System.out.println(b); System.out.println(a); } } new Inner3().test(b); } }public class Test4 { public static void main(String[] args) { LocalInner inner = new LocalInner(); inner.doSomething(4); } }
class A{ class B{} } 这样的类被称为内部类,又被称为内隐类. 从简单到深入一步一步的分析内部类的特点。 class OuterClass{ static class A{//静态内部类 public A( ){ System.out.println("Test$A !"); } } class B{//非静态内部类 public B(){ System.out.println("Test$B !"); } } public void disp( ) { final int a=10; int b; class C //成员函数中的局部内部类 { public C( ) { System.out.println(“in class C a="+a); //System.out.println(b); } } C c=new C( ); } }public class Test extends OuterClass { public static void main(String args[]) { OuterClass.A a=new OuterClass.A(); //建立静态内部类对象 B b=new OuterClass( ).new B(); //建立非静态内部类的对象 //注意这个OuterClass().new B();相当于生成一个外部类的对象,然后在利用外部类对象生成内部类对象 OuterClass t=new OuterClass( ); t.disp( ); //通过外部对象调用一个对象方法的形式,新建立了对象C. } } 注意在上面的b在运行时会为0,因为是类属性. class OuterClass { static class A { } //静态内部类 class B { } //非静态内部类 public void disp( ) { class C{ } //局部内部类 } } 编译后的将产生下面的一些类文件: OuterClass.class OuterClass$A.class OutClass$B.class OuterClass$1$C.class 记住以下几句话: 1,一个内部类的对象能够访问创建它的外部类对象的所有属性及方法(包括私有部分)。 //可以闭上眼镜,把这个内部类等同于一个类的一个方法,当然就可以访问这个外部类的 //所有方法和属性,私有方法和属性是属于外部类的,当然也就等同于内部类的. 2,对于同一个包中的其它类来说,内部类能够隐藏起来。(将内部类用private修饰即可) //只有在内部类中,才能定义一个为private类型的class,因为这时编译器已经把这个类看作这个类的成员了,但是在一般使用时,就是所谓的”顶级类时”,不能使用private,只能是public 或者是friendly. 如果要是想保证一个类不产生任何的对象,请在构造函数中,把构造函数声明成private. 3, 内部类可定义在方法中,称为局部内部类,但它只能使用方法中的final常量。 // 定义在一个方法内的类,又被成为局部内部类,这个类只能使用在方法中的final常量,注意,这个常量是在一个方法中的,那么能否使用一个类中的常量呢? 当然是可以的,因为类中的常量在在一个方法中是可见的. // 如果把一个类写在了一个if中,比如这样: class A{ int a = 10; if(a!=10){ class B{ B(){ System.out.println(a); } } } } 在编译后会有几个错误呢? 首先那个a没有被定义为final,你有一次上了圈套. 类B同样会被生成出来,只是离开了if域就失效了. 4,内部类可以被定义为抽象类 // abstract 类同样可以在内部类中 5, 非静态内部类不能声明本类的static成员 //只有一个静态的内部类,才可以声明一个static成员, class A{ static class B{ //如果这里不是一个static类,是不可以被声明这个gg方法的. static void gg(){ int a = 100; System.out.println(a); } } } class aa{ public static void main(String args[]){ A.B hh = new A.B(); hh.gg(); } } 使用内部类可以非常方便的编写事件驱动程序 这个在写事件驱动时,会有很好的解释. 匿名内部类 在某些情况下,我们只需要内部类的一个对象,那么我们就没有必要给内部类命名,没有名字的内部类我们称为匿名内部类 public class A extends Applet { public void init( ) { addMouseListener( new B( ) ); } class B extends MouseAdapter { public void mousePressed(MouseEvent me) { showStatus("Mouse Pressed."); } } } 用匿名内隐类修改: import java.applet.*; import java.awt.event.*; import java.awt.*; public class AnonymousInnerClassDemo extends Applet { public void init( ) { addMouseListener( new MouseAdapter( ) { public void mousePressed(MouseEvent me) { showStatus("Mouse Pressed"); } } ); } } 下面是一个think in java里的例子 public class Pr{ public Concents cont(){ return new Concents(){ private int i= 11; public int value (){return i;} };//这里是有一个逗号 } } 这个Contents是通过默认的构造函数产生的,匿名类的产生也就是一个新类向上转型到父类的过程. 那么如果一个父类没有一个默认的构造函数,应该什么办呢? 那就只有调用一个super()了. class noname{ Warpping wrap(int x){ return new Warpping(x){ public int value(){ return super.value()*47; } }; } public static void main(String s[]){ noname p = new noname(); Warpping w = p.wrap(10); } } 如果在匿名内部类中用到了外部对象 , 就必须保证这个外部对象是final的 。public class PP{ public DD dest(final String dest, final float price){ return new DD(){ private int cost; { cost = Math.round(price); if(cost>100) System.out.println("Over budget"); } }; } publc static void main(String []arg){ PP h = new PP(); DD d = h.dest("haha",11.111); } }
class innerA{//内部类
}
}
使用技巧:内部类和匿名类优化Java代码 http://www.leadtoit.cn/article.php?id=78
(注:所有使用内部类的地方都可以不用内部类,但使用内部类可以使程序更加的简洁,便于命名规范和划分层次结构)。
内部类是指在一个外部类的内部再定义一个类。
*内部类可为静态,可用PROTECTED和PRIVATE修饰。(而外部类不可以:顶级类只能使用PUBLIC和DEFAULT)。
*JAVA文件中没有publie class 可以类名和文件不同名。
7.11.1. 内部类的分类
成员内部类、
局部内部类、
静态内部类、
匿名内部类(图形是要用到,必须掌握)。7.11.2. 成员内部类
作为外部类的一个成员存在,与外部类的属性、方法并列。
内部类和外部类的实例变量可以共存。
在内部类中访问实例变量:this.属性
在内部类访问外部类的实例变量:外部类名.this.属性。
在外部类的外部访问内部类,使用out.inner.成员内部类的特点:
1.内部类作为外部类的成员,可以访问外部类的私有成员或属性。(即使将外部类声明为PRIVATE,但是对于处于其内部的内部类还是可见的。)
2.用内部类定义在外部类中不可访问的属性。这样就在外部类中实现了比外部类的private还要小的访问权限。
注意:内部类是一个编译时的概念,一旦编译成功,就会成为完全不同的两类。
对于一个名为outer的外部类和其内部定义的名为inner的内部类。编译完成后出现outer.class和outer$inner.class两类。
3.成员内部类不能有静态属性建立内部类对象时应注意:
在外部类的内部可以直接使用inner s=new inner();(因为外部类知道inner是哪个类,所以可以生成对象。)
而在外部类的外部,要生成(new)一个内部类对象,需要首先建立一个外部类对象(外部类可用),然后在生成一个内部类对象。
Outer o=new Outer();
Outer.Inner in=o.new.Inner()。
7.11.3. 局部内部类
在方法中定义的内部类称为局部内部类。
与局部变量类似,在局部内部类前不加修饰符public和private,其范围为定义它的代码块。注意:
局部内部类不仅可以访问外部类实例变量,但可以访问外部类的局部常量
在类外不可直接访问局部内部类(保证局部内部类对外是不可见的)。
在方法中才能调用其局部内部类。7.11.4. 静态内部类
(注意:前三种内部类与变量类似,所以可以对照参考变量)
静态内部类定义在类中,任何方法外,用static定义。
静态内部类只能访问外部类的静态成员。
生成(new)一个静态内部类不需要外部类成员:这是静态内部类和成员内部类的区别。静态内部类的对象可以直接生成:
Outer.Inner in=new Outer.Inner();
而不需要通过生成外部类对象来生成。这样实际上使静态内部类成为了一个顶级类。静态内部类不可用private来进行定义。注意:当类与接口(或者是接口与接口)发生方法命名冲突的时候,此时必须使用内部类来实现。
用接口不能完全地实现多继承,用接口配合内部类才能实现真正的多继承。
例子:
对于两个类,拥有相同的方法:
class People
{
run();
}
interface Machine{
run();
}
此时有一个robot类:
class Robot extends People implement Machine.
此时run()不可直接实现。
interface Machine
{
void run();
}
class Person
{
void run(){System.out.println("run");}
}
class Robot extends Person
{
private class MachineHeart implements Machine
{
public void run(){System.out.println("heart run");}
}
public void run(){System.out.println("Robot run");}
Machine getMachine(){return new MachineHeart();}
}
class Test
{
public static void main(String[] args)
{
Robot robot=new Robot();
Machine m=robot.getMachine();
m.run();
robot.run();
}
}
7.11.5. 匿名内部类
匿名内部类是一种特殊的局部内部类,它是通过匿名类实现接口。
IA被定义为接口。
IA I=new IA(){};
注:一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类,没有类名,根据多态,我们使用其父类名。
因其为局部内部类,那么局部内部类的所有限制都对其生效。
匿名内部类是唯一一种无构造方法类。
匿名内部类在编译的时候由系统自动起名Out$1.class。
如果一个对象编译时的类型是接口,那么其运行的类型为实现这个接口的类。
因匿名内部类无构造方法,所以其使用范围非常的有限。
class MemberInner {
private int a = 3; public class Inner2 {
private int a = 2; public void doSomething() {
// 调用外部类的属性
System.out.println(MemberInner.this.a);// 这块要注意......很重要!!!
System.out.println(a);
}
}
}public class Test3 {
public static void main(String[] args) {
MemberInner.Inner2 inner2 = new MemberInner().new Inner2();// 非静态内部类要new实例
inner2.doSomething();
}
}
2.静态内部类//静态内部类......相当于静态方法
class StaticInner{
private static int a=3;
public static class Inner{
public void test(){
System.out.println(a);
}
}
}public class Test2 {
public static void main(String[] args) {
StaticInner.Inner inner=new StaticInner.Inner();//静态内部类直接调用
inner.test();
}
}3.局部内部类 //局部内部类......相当于局部变量
class LocalInner {
public void doSomething(int b) {
final int a = 3;// 只能访问final的变量
class Inner3 {
public void test(int b) {
System.out.println(b);
System.out.println(a);
}
}
new Inner3().test(b);
}
}public class Test4 {
public static void main(String[] args) {
LocalInner inner = new LocalInner();
inner.doSomething(4);
}
}
1、真正的多重继承要用内部类来实现。
2、当你在对象的内部必须持有本对象的引用,从而可以方便的访问本对象其它成员(包括private的成员)时,用内部类。
3。当某个类只和某个实现有关,这个类的存在价值仅限于这个实现时,这个类应该定义为内部类。以上所举,不一定全面,但内部类的使用还是有一定的规律可寻的。
class B{}
}
这样的类被称为内部类,又被称为内隐类. 从简单到深入一步一步的分析内部类的特点。
class OuterClass{
static class A{//静态内部类
public A( ){
System.out.println("Test$A !");
}
}
class B{//非静态内部类
public B(){
System.out.println("Test$B !");
}
}
public void disp( ) { final int a=10; int b; class C //成员函数中的局部内部类 { public C( ) { System.out.println(“in class C a="+a); //System.out.println(b); } } C c=new C( ); } }public class Test extends OuterClass { public static void main(String args[]) {
OuterClass.A a=new OuterClass.A(); //建立静态内部类对象 B b=new OuterClass( ).new B(); //建立非静态内部类的对象 //注意这个OuterClass().new B();相当于生成一个外部类的对象,然后在利用外部类对象生成内部类对象 OuterClass t=new OuterClass( ); t.disp( ); //通过外部对象调用一个对象方法的形式,新建立了对象C. } } 注意在上面的b在运行时会为0,因为是类属性. class OuterClass { static class A { } //静态内部类 class B { } //非静态内部类 public void disp( ) { class C{ } //局部内部类 } } 编译后的将产生下面的一些类文件: OuterClass.class OuterClass$A.class OutClass$B.class OuterClass$1$C.class
记住以下几句话: 1,一个内部类的对象能够访问创建它的外部类对象的所有属性及方法(包括私有部分)。 //可以闭上眼镜,把这个内部类等同于一个类的一个方法,当然就可以访问这个外部类的 //所有方法和属性,私有方法和属性是属于外部类的,当然也就等同于内部类的. 2,对于同一个包中的其它类来说,内部类能够隐藏起来。(将内部类用private修饰即可) //只有在内部类中,才能定义一个为private类型的class,因为这时编译器已经把这个类看作这个类的成员了,但是在一般使用时,就是所谓的”顶级类时”,不能使用private,只能是public 或者是friendly. 如果要是想保证一个类不产生任何的对象,请在构造函数中,把构造函数声明成private. 3, 内部类可定义在方法中,称为局部内部类,但它只能使用方法中的final常量。 // 定义在一个方法内的类,又被成为局部内部类,这个类只能使用在方法中的final常量,注意,这个常量是在一个方法中的,那么能否使用一个类中的常量呢? 当然是可以的,因为类中的常量在在一个方法中是可见的. // 如果把一个类写在了一个if中,比如这样: class A{ int a = 10; if(a!=10){ class B{ B(){ System.out.println(a); } } } }
在编译后会有几个错误呢? 首先那个a没有被定义为final,你有一次上了圈套. 类B同样会被生成出来,只是离开了if域就失效了.
4,内部类可以被定义为抽象类 // abstract 类同样可以在内部类中 5, 非静态内部类不能声明本类的static成员 //只有一个静态的内部类,才可以声明一个static成员, class A{ static class B{ //如果这里不是一个static类,是不可以被声明这个gg方法的. static void gg(){ int a = 100; System.out.println(a); } } } class aa{ public static void main(String args[]){ A.B hh = new A.B(); hh.gg(); } }
使用内部类可以非常方便的编写事件驱动程序 这个在写事件驱动时,会有很好的解释.
匿名内部类
在某些情况下,我们只需要内部类的一个对象,那么我们就没有必要给内部类命名,没有名字的内部类我们称为匿名内部类
public class A extends Applet { public void init( ) { addMouseListener( new B( ) ); } class B extends MouseAdapter { public void mousePressed(MouseEvent me) { showStatus("Mouse Pressed."); } } } 用匿名内隐类修改:
import java.applet.*; import java.awt.event.*; import java.awt.*; public class AnonymousInnerClassDemo extends Applet { public void init( ) { addMouseListener( new MouseAdapter( ) { public void mousePressed(MouseEvent me) { showStatus("Mouse Pressed"); } } ); } }
下面是一个think in java里的例子 public class Pr{ public Concents cont(){ return new Concents(){ private int i= 11; public int value (){return i;} };//这里是有一个逗号 } } 这个Contents是通过默认的构造函数产生的,匿名类的产生也就是一个新类向上转型到父类的过程. 那么如果一个父类没有一个默认的构造函数,应该什么办呢? 那就只有调用一个super()了.
class noname{ Warpping wrap(int x){ return new Warpping(x){ public int value(){ return super.value()*47; } }; } public static void main(String s[]){ noname p = new noname(); Warpping w = p.wrap(10); } } 如果在匿名内部类中用到了外部对象 , 就必须保证这个外部对象是final的 。public class PP{ public DD dest(final String dest, final float price){ return new DD(){ private int cost; { cost = Math.round(price); if(cost>100) System.out.println("Over budget"); } }; } publc static void main(String []arg){ PP h = new PP(); DD d = h.dest("haha",11.111); } }