java中内部类使用小结

内部类的声明与访问

    内部类可以是static的也可以用其它四种访问修饰符(而外部类只能使用public和 default修饰)。内部类所能实现的功能外部类也同样可以全部实现,只是使用内部类可以使其更加简洁高效。
    内部类是一个编译时概念,一旦编译成功就会变成两个完全不同的类,对于一个名为outer的外部类和一个名为inner的内部类。编译结束之后会出现outer.class和outer$inner.class两类。所以外部类的属性/方法名可以和内部类完全一样。同时内部类可以访问外部类全部的属性和方法(包括private修饰的),但如果是具有相同命名的属性和方法则需要一定修饰:外部内名.this.外部成员名;同时如果要创建内部类对象不能直接new一个内部类,必须要通过外部类。比如要创建一个内部类iner对象,需要这么做:
    Outer outer = new Outer();
    Outer.Inner iner = outer.new Inner();
package cn.chujian6.anonymityclass;

public class InnerTest {
    public static void main(String[] args) {
            Outer outer = new Outer();
            outer.test();
            System.out.println(outer.getI());
            System.out.println("-------1--------");

            Outer.Inner iner = outer.new Inner();
            iner.innerMsg();
            System.out.println(iner.getI());
            System.out.println("-------2--------");

            System.out.println(outer.getI());
    }
}   
    class Outer {
            private int i = 10;
            private int y = 8;

            public Outer() {
                    System.out.println("调用Outer构造方法:outer");
            }

            public void sayMsg() {
                    System.out.println("Outer class!");
            }
            class Inner {
                    int i = 1000;

                    Inner() {
                            System.out.println("调用Inner构造方法:inner");
                            //可以直接访问外部类
                            System.out.println(y);
                    }
                    void innerMsg() {
                            System.out.println(">>>>>Inner class!");
                            sayMsg();
                            //访问内部类自己的成员i,也可以写成 this.i++
                            this.i++;
                            //访问外部类的成员 i和y
                            Outer.this.i++;
                            y--;
                    }
                    int getI() {
                            return i;
                    }
            }
            public void test() {
                    Inner in = new Inner();
                    in.innerMsg();
            }
            public int getI() {
                    return i;
            }
            public void setI(int i) {
                    this.i = i;
            }
    }

局部内部类

局部类不能用public或private访问说明符进行声明。它的作用域被限定在声明这个局部类块中。局部类有一个优势,他可以对外部完全隐藏起来。Thinking in java中给出了这个例子:

//定义在方法内
    public class Parcel4 { 
        public Destination destination(String s) { 
            class PDestination implements Destination { 
                private String label; 

                private PDestination(String whereTo) { 
                    label = whereTo; 
                } 

                public String readLabel() { 
                    return label; 
                } 
            } 
            return new PDestination(s); 
        } 

        public static void main(String[] args) { 
            Parcel4 p = new Parcel4(); 
            Destination d = p.destination("Tasmania"); 
        } 
    } 
    //定义在作用域内
        public class Parcel5 { 
        private void internalTracking(boolean b) { 
            if (b) { 
                class TrackingSlip { 
                    private String id; 
                    TrackingSlip(String s) { 
                        id = s; 
                    } 
                    String getSlip() { 
                        return id; 
                    } 
                } 
                TrackingSlip ts = new TrackingSlip("slip"); 
                String s = ts.getSlip(); 
            } 
        } 

        public void track() { 
            internalTracking(true); 
        } 

        public static void main(String[] args) { 
            Parcel5 p = new Parcel5(); 
            p.track(); 
        } 
    } 

局部类和普通类相似,只是作用域不同而已,它只能在该方法或者条件的作用域类有效。

静态内部类

    有的时候内部类只是为了把一个类隐藏到另外一个类的内部,并不需要内部类引用外部类的对象。因此声名为static,以便取消产生的引用。
   在内部类不需要访问外围类对象的时候,应该使用静态内部类。
   与常规类不同,静态内部类可以有静态方法和静态域。

匿名内部类

   匿名内部类是没有名称的类,所以没有办法引用他们,只能在创建的时候作为new语句的一部分来声明它。形式如下:new (类或接口){类的主体};这种形式的new语句声明的一个新的匿名内部类,它完成了对给定类的扩展或者是实现了一个给定接口,同时还创建了那个类的实例,并把它作为语句的结果返回。
   加入这个匿名类是对一个类进行扩展,它可以访问这个类的属性和方法等,和其它标准类相同;如果是实现了一个给定的接口,那就必须要实现接口的方法。注意匿名类声明是在编译时运行,实例化是在运行时进行。
   有一点需要注意,由于匿名类没有名字,所以他们没有构造方法,但是如果它继承了一个没有无参构造器的类,创建它的时候就必须带上这些参数,并且在使用过程中调用super关键字调用相应的内容。如果想初始化它的成员变量,有几种方法:
   1.如果是一个方法的匿名内部类,可以利用这个方法传进你想要的参数,这些参数必须使用final修饰;
   2.将匿名内部类改造成有名字的局部内部类,这样就可以有构造方法了;
   3.在这个匿名内部类中使用初始化块。
    public class Outer { 
        public static void main(String[] args) { 
            Outer outer = new Outer(); 
            Inner inner = outer.getInner("Inner", "gz"); 
            System.out.println(inner.getName()); 
        } 
      //这里的city并没有被使用,所以不需要final
        public Inner getInner(final String name, String city) { 
            return new Inner(name, city) { 
                private String nameStr = name; 
                public String getName() { 
                    return nameStr; 
                } 
            }; 
        } 
    } 
    abstract class Inner { 
        Inner(String name, String city) { 
            System.out.println(city); 
        } 

        abstract String getName(); 
    } 
 留意外部类的方法的形参,当所在的方法的形参需要被内部类里面使用时,该形参必须为final。这里可以看到形参name已经定义为final了,而形参city 没有被使用则不用定义为final。为什么要定义为final呢?在网上找到的解释:
“这是一个编译器设计的问题,如果你了解java的编译原理的话很容易理解。首先,内部类被编译的时候会生成一个单独的内部类的.class文件,这个文件并不与外部类在同一class文件中。当外部类传的参数被内部类调用时,从java程序的角度来看是直接的调用例如:  
public void dosome(final String a,final int b){  
  class Dosome{public void dosome(){System.out.println(a+b)}};  
  Dosome some=new Dosome();  
  some.dosome();  
}  
从代码来看好像是那个内部类直接调用的a参数和b参数,但是实际上不是,在java编译器编译以后实际的操作代码是  
class Outer$Dosome{  
  public Dosome(final String a,final int b){  
  this.Dosome$a=a;  
  this.Dosome$b=b;  
}  
  public void dosome(){  
  System.out.println(this.Dosome$a+this.Dosome$b);  
}  
}}  
 从以上代码看来,内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到了自己的内部,自己内部的方法调用的实际是自己的属性而不是外部类方法的参数。这样理解就很容易得出为什么要用final了,因为两者从外表看起来是同一个东西,实际上却不是这样,如果内部类改掉了这些参数的值也不可能影响到原参数,然而这样却失去了参数的一致性,因为从编程人员的角度来看他们是同一个东西,如果编程人员在程序设计的时候在内部类中改掉参数的值,但是外部调用的时候又发现值其实没有被改掉,这就让人非常的难以理解和接受,为了避免这种尴尬的问题存在,所以编译器设计人员把内部类能够使用的参数设定为必须是final来规避这种莫名其妙错误的存在。” (简单理解就是,拷贝引用,为了避免引用值发生改变,例如被外部类的方法修改等,而导致内部类得到的值不一致,于是用final来让该引用不可改变)。

 因为匿名内部类,没名字,是用默认的构造函数的,无参数的,那如果需要参数呢?则需要该类有带参数的构造函数:
public class Outer { 
        public static void main(String[] args) { 
            Outer outer = new Outer(); 
            Inner inner = outer.getInner("Inner", "gz"); 
            System.out.println(inner.getName()); 
        } 

        public Inner getInner(final String name, String city) { 
            return new Inner(name, city) { 
                private String nameStr = name; 

                public String getName() { 
                    return nameStr; 
                } 
            }; 
        } 
    } 
    abstract class Inner { 
        Inner(String name, String city) { 
            System.out.println(city); 
        } 
        abstract String getName(); 
    } 
注意这里的形参city,由于它没有被匿名内部类直接使用,而是被抽象类Inner的构造函数所使用,所以不必定义为final。

而匿名内部类通过实例初始化,可以达到类似构造器的效果:
 public class Outer { 
        public static void main(String[] args) { 
            Outer outer = new Outer(); 
            Inner inner = outer.getInner("Inner", "gz"); 
            System.out.println(inner.getName()); 
            System.out.println(inner.getProvince()); 
        } 

        public Inner getInner(final String name, final String city) { 
            return new Inner() { 
                private String nameStr = name; 
                private String province; 

                // 实例初始化 
                { 
                    if (city.equals("gz")) { 
                        province = "gd"; 
                    }else { 
                        province = ""; 
                    } 
                } 

                public String getName() { 
                    return nameStr; 
                } 

                public String getProvince() { 
                    return province; 
                } 
            }; 
        } 
    } 

    interface Inner { 
        String getName(); 
        String getProvince(); 
    } 

【解惑】领略Java内部类的“内部”

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值