java编程思想阅读之内部类

来源:互联网 发布:影视大全软件下载 编辑:程序博客网 时间:2024/05/16 18:07
实例1:内部类的基本结构
//外部类class Out {    private int age = 12;         //内部类    class In {        public void print() {            System.out.println(age);        }    }} public class Demo {    public static void main(String[] args) {        Out.In in = new Out().new In();        in.print();        //或者采用下种方式访问        /*        Out out = new Out();        Out.In in = out.new In();        in.print();        */    }}
运行结果:12

从上面的例子不难看出,内部类其实严重破坏了良好的代码结构,但为什么还要使用内部类呢?
因为内部类可以随意使用外部类的成员变量(包括私有)而不用生成外部类的对象,这也是内部类的唯一优点
如同心脏可以直接访问身体的血液,而不是通过医生来抽血


程序编译过后会产生两个.class文件,分别是Out.class和Out$In.class
其中$代表了上面程序中Out.In中的那个 .
Out.In in = new Out().new In()可以用来生成内部类的对象,这种方法存在两个小知识点需要注意
  1.开头的Out是为了标明需要生成的内部类对象在哪个外部类当中

  2.必须先有外部类的对象才能生成内部类的对象,因为内部类的作用就是为了访问外部类中的成员变量

实例2:内部类中的变量访问形式

class Out {    private int age = 12;         class In {        private int age = 13;        public void print() {            int age = 14;            System.out.println("局部变量:" + age);            System.out.println("内部类变量:" + this.age);            System.out.println("外部类变量:" + Out.this.age);        }    }} public class Demo {    public static void main(String[] args) {        Out.In in = new Out().new In();        in.print();    }}
运行结果:
局部变量:14
内部类变量:13
外部类变量:12

从实例1中可以发现,内部类在没有同名成员变量和局部变量的情况下,内部类会直接访问外部类的成员变量,而无需指定Out.this.属性名
否则,内部类中的局部变量会覆盖外部类的成员变量
而访问内部类本身的成员变量可用this.属性名,访问外部类的成员变量需要使用Out.this.属性名
实例3:静态内部类
class Out {    private static int age = 12;         static class In {        public void print() {            System.out.println(age);        }    }} public class Demo {    public static void main(String[] args) {        Out.In in = new Out.In();        in.print();    }}
运行结果:12

可以看到,如果用static 将内部内静态化,那么内部类就只能访问外部类的静态成员变量,具有局限性
其次,因为内部类被静态化,因此Out.In可以当做一个整体看,可以直接new 出内部类的对象(通过类名访问static,生不生成外部类对象都没关系)
实例4:私有内部类

class Out {    private int age = 12;         private class In {        public void print() {            System.out.println(age);        }    }    public void outPrint() {        new In().print();    }} public class Demo {    public static void main(String[] args) {        //此方法无效        /*        Out.In in = new Out().new In();        in.print();        */        Out out = new Out();        out.outPrint();    }}
运行结果:12
如果一个内部类只希望被外部类中的方法操作,那么可以使用private声明内部类
上面的代码中,我们必须在Out类里面生成In类的对象进行操作,而无法再使用Out.In in = new Out().new In() 生成内部类的对象
也就是说,此时的内部类只有外部类可控制
如同是,我的心脏只能由我的身体控制,其他人无法直接访问它
实例5:方法内部类
class Out {    private int age = 12;     public void Print(final int x) {        class In {            public void inPrint() {                System.out.println(x);                System.out.println(age);            }        }        new In().inPrint();    }} public class Demo {    public static void main(String[] args) {        Out out = new Out();        out.Print(3);    }}
运行结果:

3
12

在上面的代码中,我们将内部类移到了外部类的方法中,然后在外部类的方法中再生成一个内部类对象去调用内部类方法
如果此时我们需要往外部类的方法中传入参数,那么外部类的方法形参必须使用final定义
至于final在这里并没有特殊含义,只是一种表示形式而已
实例6:匿名内部类

abstract class Bird {      private String name;        public String getName() {          return name;      }        public void setName(String name) {          this.name = name;      }            public abstract int fly();  }    public class Test {            public void test(Bird bird){          System.out.println(bird.getName() + "能够飞 " + bird.fly() + "米");      }            public static void main(String[] args) {          Test test = new Test();          test.test(new Bird() {                            public int fly() {                  return 10000;              }                            public String getName() {                  return "大雁";              }          });      }  }  ------------------  Output:  大雁能够飞 10000米  
在Test类中,test()方法接受一个Bird类型的参数,同时我们知道一个抽象类是没有办法直接new的,我们必须要先有实现类才能new出来它的实现类实例。所以在mian方法中直接使用匿名内部类来创建一个Bird实例。
       由于匿名内部类不能是抽象类,所以它必须要实现它的抽象父类或者接口里面所有的抽象方法。
       对于这段匿名内部类代码其实是可以拆分为如下形式:
public class WildGoose extends Bird{      public int fly() {          return 10000;      }            public String getName() {          return "大雁";      }  }  
WildGoose wildGoose = new WildGoose();  test.test(wildGoose);  
 在这里系统会创建一个继承自Bird类的匿名类的对象,该对象转型为对Bird类型的引用。
       对于匿名内部类的使用它是存在一个缺陷的,就是它仅能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立即消失,所以匿名内部类是不能够被重复使用。对于上面的实例,如果我们需要对test()方法里面内部类进行多次使用,建议重新定义类,而不是使用匿名内部类。
注意事项:匿名内部类

 在使用匿名内部类的过程中,我们需要注意如下几点:
      1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口。
      2、匿名内部类中是不能定义构造函数的。
      3、匿名内部类中不能存在任何的静态成员变量和静态方法。
      4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。
      5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

原创粉丝点击