zoukankan      html  css  js  c++  java
  • Java 内部类

    在Java中,定义在一个类里面或者一个方法里面的类 叫 内部类,包含内部类的类称为外部类。广泛意义上的内部类一般包括四种:成员内部类局部内部类匿名内部类静态内部类


    1.成员内部类
    (1)该类像是外部类的一个成员,可以无条件的访问外部类的所有成员属性和成员方法(包括private成员和静态成员);
    (2)成员内部类拥有与外部类同名的成员变量时,会发生隐藏现象,即默认情况下访问的是成员内部类中的成员。如果要访问外部类中的成员,需要以下形式访问:【外部类.this.成员变量 或 外部类.this.成员方法】;
    (3)在外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问;
    (4)成员内部类是依附外部类而存在的,也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象;
    (5)内部类可以拥有private访问权限、protected访问权限、public访问权限及包访问权限。如果成员内部类用private修饰,则只能在外部类的内部访问;如果用public修饰,则任何地方都能访问;如果用protected修饰,则只能在同一个包下或者继承外部类的情况下访问;如果是默认访问权限,则只能在同一个包下访问。外部类只能被public和包访问两种权限修饰。


    2.局部内部类
    (1)局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内;
    (2)局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。


    3.匿名内部类
    (1)一般使用匿名内部类的方法来编写事件监听代码;
    (2)匿名内部类是不能有访问修饰符和static修饰符的;
    (3)匿名内部类是唯一一种没有构造器的类;
    (4)匿名内部类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的实现或是重写。


    4.内部静态类
    (1)静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似;
    (2)不能使用外部类的非static成员变量或者方法。

    内部类(Inner Class)和静态内部类(Static Nested Class)的区别:

    内部类可以声明public、protected、private等访问限制,可以声明 为abstract的供其他内部类或外部类继承与扩展,或者声明为static、final的,也可以实现特定的接口。外部类按常规的类访问方式使用内部 类,唯一的差别是外部类可以访问内部类的所有方法与属性,包括私有方法与属性

    (1)创建实例

       OutClass.InnerClass obj = outClassInstance.new InnerClass(); //注意是外部类实例.new,内部类

       AAA.StaticInner in = new AAA.StaticInner();//注意是外部类本身,静态内部类

    (2)内部类中的this

       内 部类中的this与其他类一样是指的本身。创建内部类对象时,它会与创造它的外围对象有了某种联系,于是能访问外围类的所有成员,不需任何特殊条件,可理 解为内部类链接到外部类。 用外部类创建内部类对象时,此内部类对象会秘密的捕获一个指向外部类的引用,于是,可以通过这个引用来访问外围类的成员。

    (3)外部类访问内部类

       内部类类似外部类的属性,因此访问内部类对象时总是需要一个创建好的外部类对象。内部类对象通过‘外部类名.this.xxx’的形式访问外部类的属性与方法。如:
           System.out.println("Print in inner Outer.index=" + pouter.this.index);
           System.out.println("Print in inner Inner.index=" + this.index);

    (4)内部类向上转型

       内部类也可以和普通类一样拥有向上转型的特性。将内部类向上转型为基类型,尤其是接口时,内部类就有了用武之地。如果内部类是private的,只可以被它的外部类问,从而完全隐藏实现的细节。

    (5)方法内的类

       方法内创建的类(注意方法中也能定义类),不能加访问修饰符。另外,方法内部的类也不是在调用方法时才会创建的,它们一样也被事先编译了。

    (6)静态内部类

       定义静态内部类:在定义内部类的时候,可以在其前面加上一个权限修饰符static。此时这个内部类就变为了静态内部类。

    通常称为嵌套类,当内部类是static时,意味着:

       [1]要创建嵌套类的对象,并不需要其外围类的对象;

       [2]不能从嵌套类的对象中访问非静态的外围类对象(不能够从静态内部类的对象中访问外部类的非静态成员);

       嵌 套类与普通的内部类还有一个区别:普通内部类的字段与方法,只能放在类的外部层次上,所以普通的内部类不能有static数据和static字段, 也不能包含嵌套类。但是在嵌套类里可以包含所有这些东西。也就是说,在非静态内部类中不可以声明静态成员,只有将某个内部类修饰为静态类,然后才能够在这 个类中定义静态的成员变量与成员方法。

       另外,在创建静态内部类时不需要将静态内部类的实例绑定在外部类的实例上。普通非静态内部类的 对象是依附在外部类对象之中的,要在一个外部类中定义一个静态的内部类,不需要利用关键字new来创建内部类的实例。静态类和方法只属于类本身,并不属于 该类的对象,更不属于其他外部类的对象。

    (7)内部类标识符

       每个类会产生一个.class文件,文件名即为类名。同样,内部类也会产生这么一个.class文件,但是它的名称却不是内部类的类名,而是有着严格的限制:外围类的名字,加上$,再加上内部类名字。

    (8)为何要用内部类?

       1.   内部类一般只为其外部类使用;

       2.   内部类提供了某种进入外部类的窗户;

       3.   也是最吸引人的原因,每个内部类都能独立地继承一个接口,而无论外部类是否已经继承了某个接口。因此,内部类使多重继承的解决方案变得更加完整。

    加深印象,参考一下:

    package com.test.xml;
    public class OutClassTest {
        static int a;

        int b;

        public static void test() {
            System.out.println("outer class static function");
        }

        public static void main(String[] args) {
            OutClassTest oc = new OutClassTest();
            // new一个外部类
            OutClassTest oc1 = new OutClassTest();
            // 通过外部类的对象new一个非静态的内部类
            OutClassTest.InnerClass no_static_inner = oc1.new InnerClass();
            // 调用非静态内部类的方法
            System.out.println(no_static_inner.getKey());

            // 调用静态内部类的静态变量
            System.out.println(OutClassTest.InnerStaticClass.static_value);
            // 不依赖于外部类实例,直接实例化内部静态类
            OutClassTest.InnerStaticClass inner = new OutClassTest.InnerStaticClass();
            // 调用静态内部类的非静态方法
            System.out.println(inner.getValue());
            // 调用内部静态类的静态方法
            System.out.println(OutClassTest.InnerStaticClass.getMessage());
        }

        private class InnerClass {
            // 只有在静态内部类中才能够声明或定义静态成员
            // private static String tt = "0";
            private int flag = 0;

            public InnerClass() {
                // 三.非静态内部类的非静态成员可以访问外部类的非静态变量和静态变量
                System.out.println("InnerClass create a:" + a);
                System.out.println("InnerClass create b:" + b);
                System.out.println("InnerClass create flag:" + flag);
                //
                System.out.println("InnerClass call outer static function");
                // 调用外部类的静态方法
                test();
            }

            public  String getKey() {
                return "no-static-inner";
            }
        }

        private static class InnerStaticClass {
            // 静态内部类可以有静态成员,而非静态内部类则不能有静态成员。
            private static String static_value = "0";

            private int flag = 0;

            public InnerStaticClass() {
                System.out.println("InnerClass create a:" + a);
                // 静态内部类不能够访问外部类的非静态成员
                // System.out.println("InnerClass create b:" + b);
                System.out.println("InnerStaticClass flag is " + flag);
                System.out.println("InnerStaticClass tt is " + static_value);
            }

            public int getValue() {
                // 静态内部类访问外部类的静态方法
                test();
                return 1;
            }

            public static String getMessage() {
                return "static-inner";
            }
        }

        public OutClassTest() {
            // new一个非静态的内部类
            InnerClass ic = new InnerClass();
            System.out.println("OuterClass create");
        }

    }

    /**
     * 总结: 
     * 1.静态内部类可以有静态成员(方法,属性),而非静态内部类则不能有静态成员(方法,属性)。
     * 2.静态内部类只能够访问外部类的静态成员,而非静态内部类则可以访问外部类的所有成员(方法,属性)。
     * 3.实例化一个非静态的内部类的方法:
     *  a.先生成一个外部类对象实例
     *  OutClassTest oc1 = new OutClassTest();
     *  b.通过外部类的对象实例生成内部类对象
     *  OutClassTest.InnerClass no_static_inner = oc1.new InnerClass();
     *  4.实例化一个静态内部类的方法:
     *  a.不依赖于外部类的实例,直接实例化内部类对象
     *  OutClassTest.InnerStaticClass inner = new OutClassTest.InnerStaticClass();
     *  b.调用内部静态类的方法或静态变量,通过类名直接调用
     *  OutClassTest.InnerStaticClass.static_value
     *  OutClassTest.InnerStaticClass.getMessage()
     */

  • 相关阅读:
    BZOJ1070[SCOI2007]修车
    BZOJ1061[Noi2008] 志愿者招募
    BZOJ 3511 土地划分
    BZOJ3130 [Sdoi2013]费用流
    POJ1797 Heavy Transportation
    P2866 糟糕的一天
    P1155 双栈排序
    P1027 car的旅行路线
    POJ3037 Skiing
    POJ1125 Stockbroker Grapevine
  • 原文地址:https://www.cnblogs.com/lucky1024/p/10983364.html
Copyright © 2011-2022 走看看