day 15-内部类-枚举

25 篇文章 0 订阅
7 篇文章 0 订阅
  1. 类中的定义的成员:字段 方法 内部类

    为什么使用内部类:
    1):增强封装,把内部类隐藏在外部类之内,不许其他类访问内部类。
    2):内部类可以直接访问外部类的成员。

    内部类根据使用不同的修饰符或者定义的位置不同,分成四种:(类似静态成员变量,实例成员变量,局部变量)
    ① 实例内部类:  内部类没有使用static修饰.
    ② 静态内部类:  内部类使用了static修饰.
    ③ 局部内部类:  在方法中定义的内部类.
    ④ 匿名内部类适合于仅使用一次使用的类,属于局部内部类的特殊情况:

    对于每个内部类来说:Java编译器会生成独立.class文件.

    成员内部类:外部类名$内部类名字
    局部内部类:外部类名$数字内部类名称
    匿名内部类:外部类名$数字

  2. 外部类的访问修饰符:要么使用public,要么就缺省.
    内部类看做是外部类的一个成员,那么内部类可以使用public/缺省/protected/private修饰.
    还可以是static修饰.
    -------------------------------------
    实例内部类:
    没有使用static修饰内部类,说明内部类属于外部类的对象,不属于外部类本身.
    特点:
    1:创建实例内部类前,必须存在外部类对象,通过外部类对象创建内部类对象(当存在内部类对象时,一定存在外部类对象).    
            Outter.Inner in = new Outter().new Inner();
    2:实例内部类的实例自动持有外部类的实例的引用,内部类可以直接访问外部类成员.
    3:外部类中不能直接访问内部类的成员,必须通过内部类的实例去访问.
    4:实例内部类中不能定义静态成员,只能定义实例成员.
    5:如果实例内部类和外部类存在同名的字段或方法abc,那么在内部类中:
            this.abc:表示访问内部类成员.
            外部类.this.abc:表示访问外部类成员.

  3. 静态内部类:
    使用static修饰的内部类.
    特点:
    1):静态内部类的实例不会自动持有外部类的特定实例的引用,在创建内部类的实例时,不必创建外部类的实例.
         Outter.Inner in = new Outter.Inner();
    2):静态内部类可以直接访问外部类的静态成员,如果访问外部类的实例成员,必须通过外部类的实例去访问.
    3):在静态内部类中可以定义静态成员和实例成员.
    4):测试类可以通过完整的类名直接访问静态内部类的静态成员.

  4. 局部内部类(打死都不用):
    在方法中定义的内部类,其可见范围是当前方法,和局部变量是同一个级别.
    1):不能使用public,private,protected,static修饰符.
    2):局部内部类只能在当前方法中使用.
    3):局部内部类和实例内部类一样,不能包含静态成员.
    4):局部内部类和实例内部类,可以访问外部类的所有成员.
    5):局部内部类访问的局部变量必须使用final修饰(在Java8中是自动隐式加上final,但是依然是常量,不能改变值).
       原因:如果当前方法不是main方法,那么当前方法调用完毕之后,当前方法的栈帧被销毁,方法内部的局部变量的空间全部销毁.
                然后局部内部类是定义在方法中的,而且在方法中会创建局部内部类对象,而局部内部类会去访问局部变量,当当前方法被销毁的时候,对象还在堆内存,依然持有对局部变量的引用,但是方法被销毁的时候局部变量以及被销毁了.
               此时出现:在堆内存中,一个对象引用着一个不存在的数据. 为了避免该问题,我们使用final修饰局部变量,从而变成常量,永驻内存空间,即使方法销毁之后,该局部变量也在内存中,对象可以继续持有.

  5. 内部类只能访问final修饰的局部变量,外部类所有成员。 位置:方法体里,方法参数列表里
    匿名内部类(Anonymous),是一个没有名称的局部内部类,适合只使用一次的类。
    特点:
    1):匿名内部类本身没有构造器,但是会调用父类构造器.
    2):匿名类尽管没有构造器,但是可以在匿名类中提供一段初始化代码块,JVM在调用父类构造器后,会执行该段代码.
    3):内部类除了可以继承类之外,还可以实现接口.
    格式:
         new 父类构造器([实参列表]) 或 接口()
         {
             //匿名内部类的类体部分
          }

    注意:匿名内部类必须继承一个父类或者实现一个接口,但最多只能一个父类或实现一个接口。.

  6. 是一个特殊的类,就是固定的多个常量对象的集合.

    定义格式:
    [修饰符] enum  枚举类名
    {
          常量A,常量B,常量C;
    }
    -------------------------------------------------------------------
    我们自定义的枚举类型,在编译后(在底层)都是直接继承于java.lang.Enum类的,Enum是所有枚举的父类

  7. 枚举类的特点:
    枚举特点:
    枚举的直接父类java.lang.Enum,但是不能显示继承Enum。
    ② 枚举就相当于一个类,可以定义构造方法、成员变量、普通方法和抽象方法。
    默认私有的造方法,即使不写访问权限也是private(假构造器,底层没有无参数构造器)。
    每个实例分别用一个全局常量表示,枚举类的对象是固定的,实例个数有限,不能使用new关键字。
    枚举实例必须位于枚举体中的最开始部分,枚举实例列表的后要有分号与其他成员相分隔。
    ⑥ 枚举实例后有花括号时,该实例是枚举类的匿名内部类对象(查看编译后的class文件)。

  8. 枚举的使用:
       1):枚举中都是全局公共的静态常量,可以直接使用枚举类名调用.
              Weekday day = Weekday.SATURDAY;
       2):因为java.lang.Enum类是所有枚举类的父类,所以所有的枚举对象可以调用Enum类中的方法.
             String   name = 枚举对象.name();  //返回枚举对象的常量名称
             int      ordinal = 枚举对象.ordinal();//返回枚举对象的序号,从0开始.
             String  str       =  枚举对象.toString():返回 枚举对象的常量名称
       3):编译器生成的枚举类的静态方法(从反编译代码中):
             枚举类型[]  values();
              Weekday[] ws   = Weekday.values();//返回当前枚举类型所有的常量
             枚举类型  valueof(String name);
              Weekday  day = Weekday.valueOf("MONDAY");//把一个指定名称字符串转换为当前枚举类中同名的常量.
        4):switch只支持int类型,支持枚举是因为底层使用的枚举对象的ordinal,而ordinal的类型依然是int类型.

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值