特殊类

内部类的概念和分类
内部类的基本概念
  • 当一个类的定义出现在另外一个类的类体中时,那么这个类叫做内部类(Inner),而这个内部类所在的类叫做外部类(Outer)
  • 类中的内容:成员变量、成员方法、构造方法、静态成员、构造块和静态代码块、内部类
实际作用
  • 当一个类存在的价值仅仅是为某一个类单独服务时,那么就可以将这个类定义为所服务类中的内部类,这样可以隐藏该类的实现细节并且可以方便的访问外部类的私有成员而不再需要提供公有的get和set方法
内部类的分类
  • 普通内部类:直接将一个类的定义放在另外一个类的类体中
  • 静态内部类:使用static关键字修饰内部类,隶属于类层级
  • 局部内部类:直接将一个类的定义放在方法体的内部时
  • 匿名内部类:就是指没有名字的内部类
普通内部类的定义
普通(成员)内部类的格式
  • 访问修饰符 class 外部类的类名{

    ​ 访问修饰符 class 外部类的类名{

    ​ 内部类的类体

    ​ }

    }

NormalOuter.java

package com.lagou.task10;

/**
 * 编程实现普通内部类的定义和使用       - 文档注释
 */
public class NormalOuter {
    private int cnt = 1;

    // 定义普通内部类,隶属于外部类的成员,并且是对象层级
    public class NormalInner{
        private int ia = 2;

        public NormalInner(){
            System.out.println("普通内部类的构造方法执行到了!");
        }
        public void show(){
            System.out.println("外部类中的变量cnt的数值为:" + cnt);
            System.out.println("ia = " + ia);
        }
    }
}

NormalOuterTest.java

package com.lagou.task10;

public class NormalOuterTest {
    public static void main(String[] args) {

        // 1.声明NormalOuter类型的引用指向该类型的对象
        NormalOuter no = new NormalOuter();
        // 2.声明NormalOuter类中内部类的引用指向内部类的对象
        NormalOuter.NormalInner ni = no.new NormalInner();
        // 调用内部类中的show方法
        ni.show();
    }
}
普通内部类的使用方式
  • 普通内部类和普通类一样可以定义成员变量、成员变量以及构造方法
  • 普通内部类和普通类一样可以使用final或者abstract关键字修饰
  • 普通内部类还可以使用private或protected关键字修饰
  • 普通内部类需要使用外部类对象创建对象
  • 如果内部类访问外部类中与本类内部同名的成员变量或方法时,需要使用this关键字

在这里插入图片描述

静态内部类的定义
  • 访问修饰符 class 外部类的类名{

    ​ 访问修饰符 static class 外部类的类名{

    ​ 内部类的类体

    ​ }

    }

StaticOuter.java

package com.lagou.task10;

/**
 * 实现静态内部类的定义和使用
 */
public class StaticOuter {
    private int cnt = 1;     // 隶属于对象层级
    private static int snt = 2; // 隶属于类层级

    /**
     * 定义静态内部类 有static关键字修饰隶属于类层级
     */
    public static class StaticInner{
        private int ia = 3;

        public StaticInner(){
            System.out.println("静态内部类的构造方法哦!");
        }
        public void show(){
            System.out.println("ia = " + ia); // 3
            System.out.println("外部类中的snt = " + snt); // 2
            //System.out.println("外部类的cnt = " + cnt);  // Error:静态上下文中不能访问非静态的成员,因为此时可能还没有创建对象
        }
    }
}

StaticOuterTest.java

package com.lagou.task10;

public class StaticOuterTest {

    public static void main(String[] args) {

        // 1.声明StaticInner类型的引用指向该类型的对象
        StaticOuter.StaticInner si = new StaticOuter.StaticInner();
        // 2.调用show方法进行测试
        si.show();
    }
}
静态内部类的使用方式
  • 静态内部类不能直接访问外部类的非静态成员
  • 静态内部类可以直接创建对象
  • 如果静态内部类访问外部类中与本类内同名的成员变量或方法时,需要使用类名.的方式访问

StaticOuter.java

package com.lagou.task10;

/**
 * 实现静态内部类的定义和使用
 */
public class StaticOuter {
    private int cnt = 1;     // 隶属于对象层级
    private static int snt = 2; // 隶属于类层级

    public /*static*/ void show(){
        System.out.println("外部类的show方法就是这里!");
    }
    /**
     * 定义静态内部类 有static关键字修饰隶属于类层级
     */
    public static class StaticInner{
        private int ia = 3;
        private static int snt = 4;

        public StaticInner(){
            System.out.println("静态内部类的构造方法哦!");
        }
        public void show(){
            System.out.println("ia = " + ia); // 3
            System.out.println("外部类中的snt = " + snt); // 2
            //System.out.println("外部类的cnt = " + cnt);  // Error:静态上下文中不能访问非静态的成员,因为此时可能还没有创建对象
        }
        public void show2(int snt){ //就近原则
            System.out.println("snt = " + snt);
            System.out.println("内部类中的成员snt = " + StaticInner.snt); // 4
            System.out.println("外部类中的成员snt = " + StaticOuter.snt); // 2
            //StaticOuter.show();
            new StaticOuter().show();
        }
    }
}

在这里插入图片描述

局部内部类的定义
局部(方法)内部类的格式
  • 访问修饰符 class 外部类的类名{

    ​ 访问修饰符 返回值类型 成员方法名(形参列表){

    ​ class 内部类的类名{

    ​ 内部类的类体;

    ​ }

    ​ }

    }

AreaOuter.java

package com.lagou.task10;
/**
 * 编程实现局部内部类的定义和使用
 */
public class AreaOuter {
    private int cnt = 1;

    public void show(){
        // 定义局部内部类,只在当前方法体的内部好使
        class AreaInner{
            private int ia = 2;

            public AreaInner(){
                System.out.println("局部内部类的构造方法!");
            }
            public void test(){
                System.out.println("ia = " + ia); // 1
                System.out.println("cnt = " + cnt); // 2
            }
        }
        // 声明局部内部类的引用指向局部内部类的对象
        AreaInner ai = new AreaInner();
        ai.test();
    }
}

AreaOuterTest.java

package com.lagou.task10;

public class AreaOuterTest {

    public static void main(String[] args) {

        // 1.声明外部类类型的引用指向外部类的对象
        AreaOuter ao = new AreaOuter();
        // 2.通过show方法调用实现局部内部类的定义和使用
        ao.show();
    }
}
局部内部类的使用方式
  • 局部内部类只能在该方法的内部可以使用
  • 局部内部类可以在方法体内部直接创建对象
  • 局部内部类不能使用访问控制符和static关键字修饰符
  • 局部内部类可以使用外部方法的局部变量,但是必须是final的,由局部内部类和局部变量的声明周期不同所致

在这里插入图片描述

回调模式的概念和编程
回调模式的概念和编程
  • 回调模式是指—如果一个方法的参数是接口类型,则在调用该方法时,需要创建并传递一个实现此接口类型的对象;而方法在运行时会调用到参数对象中所实现的方法(接口中定义的)

AnonymousInterface.java

package com.lagou.task10;

public class AnonymousInterfaceImpl implements AnonymousInterface {
    @Override
    public void show() {
        System.out.println("这里是接口的实现类!");
    }
}

AnonymousInterfaceTest.java

package com.lagou.task10;

public class AnonymousInterfaceTest {

    // 假设已有下面的方法,请问如何调用下面的方法
    // AnonymousInterface ai = new AnonymousInterfaceImpl();
    // 接口类型的引用指向实现类型的对象,形成了多态
    public static void test(AnonymousInterface ai){
        //编译阶段调用父类版本,运行调用实现类重写的版本
        ai.show();
    }
    public static void main(String[] args) {

        //AnonymousInterfaceTest.test(new AnonymousInterface());  // Error:接口不能实例化
        AnonymousInterfaceTest.test(new AnonymousInterfaceImpl());
    }
}

AnonymousInterfaceImpl.java

package com.lagou.task10;

public class AnonymousInterfaceImpl implements AnonymousInterface {
    @Override
    public void show() {
        System.out.println("这里是接口的实现类!");
    }
}
匿名内部类的使用
匿名内部类的语法格式(重点
  • 接口/父类类型 引用变量名 = new 接口/父类类型(){方法的重写};
开发经验分享
  • 当接口/类类型的引用作为方法的形参时,实参的传递方式有两种:
  • 自定义类实现接口/继承类并重写方法,然后创建该类对象作为实参传递;
  • 使用上述匿名内部类的语法格式得到接口/类类型的引用即可

AnonymousInterfaceTest.java

package com.lagou.task10;

public class AnonymousInterfaceTest {

    // 假设已有下面的方法,请问如何调用下面的方法
    // AnonymousInterface ai = new AnonymousInterfaceImpl();
    // 接口类型的引用指向实现类型的对象,形成了多态
    public static void test(AnonymousInterface ai){
        //编译阶段调用父类版本,运行调用实现类重写的版本
        ai.show();
    }
    public static void main(String[] args) {

        //AnonymousInterfaceTest.test(new AnonymousInterface());  // Error:接口不能实例化
        AnonymousInterfaceTest.test(new AnonymousInterfaceImpl());

        System.out.println("---------------------------");
        // 使用匿名内部类的语法格式来得到接口类型的引用,格式为:接口/父类类型 引用变量名 = new 接口/父类类型(){方法的重写};
        AnonymousInterface ait = new AnonymousInterface() {
            @Override
            public void show() {
                System.out.println("匿名内部类就是这么玩的,虽然你很抽象!");
            }
        };
        AnonymousInterfaceTest.test(ait);

        System.out.println("---------------------------");
        // 从java8开始提出新特性lamda表达式可以简化上述代码,格式为:(参数列表 -> {方法体})
        AnonymousInterface ait2 = () -> System.out.println("lamda表达式原来是如此简单!");
        AnonymousInterfaceTest.test(ait2);
    }
}

在这里插入图片描述

枚举的概念和自定义实现
枚举的基本概念
  • 一年中的所有季节:春季、夏季、秋季、冬季
  • 所有的性别:男、女
  • 键盘上的所有方向案件:向上、向下、向左、向右
  • 在日常生活中这些食物的取值只有明确的几个固定值,此时描述这些事物的所有值都可以一一列举出来,而这个列举出来的类型就叫做枚举类型

Direction.java

package com.lagou.task10;

/**
 * 编程实现所有方向的枚举,所有的方向:向上、向下、向左、向右
 */
public class Direction {
    private final String desc; // 用于描述方向字符串的成员变量

    // 2.声明本类类型的引用指向本类类型的对象
    public static final Direction UP = new Direction("向上");
    public static final Direction DOWN = new Direction("向下");
    public static final Direction LEFT = new Direction("向左");
    public static final Direction RIGHT = new Direction("向右");

    // 通过构造方法实现成员变量的初始化,更加灵活
    // 1.私有化构造方法,此时该构造方法只能在本类的内部使用
    private Direction(String desc){
        this.desc = desc;
    }

    // 通过公有的get方法可以在本类的外部访问该类的成员变量的数值
    public String getDesc() {
        return desc;
    }
}

DirectionTest.java

package com.lagou.task10;

public class DirectionTest {
    public static void main(String[] args) {

        // 1.声明Direction类型的引用指向该类型的对象并打印特征
        /*Direction d1 = new Direction("向上");
        System.out.println("获取到的字符串是:" + d1.getDesc());  // 向上

        Direction d2 = new Direction("向下");
        System.out.println("获取到的字符串是:" + d1.getDesc());  // 向下

        Direction d3 = new Direction("向左");
        System.out.println("获取到的字符串是:" + d3.getDesc());  // 向左

        Direction d4 = new Direction("向右");
        System.out.println("获取到的字符串是:" + d4.getDesc());  // 向右

        System.out.println("-----------------------------");
        Direction d5 = new Direction("向前");
        System.out.println("获取到的字符串是:" + d5.getDesc());  // 向前
        */
        //Direction.up = 2; Error:类型不匹配
        //Direction d2 = null;
        //Direction.UP = d2; Error:final关键字修饰
        Direction d1 = Direction.UP;
        Direction d2 = Direction.DOWN;
        Direction d3 = Direction.LEFT;
        Direction d4 = Direction.RIGHT;
        System.out.println("获取到的方向是:" + d1.getDesc() + "、" + d2.getDesc() + "、" + d3.getDesc() + "、" + d4.getDesc());



    }
}

在这里插入图片描述

枚举类型的定义
  • 使用public static final表示的常量描述较为繁琐,使用enum关键字来定义枚举类型取代常量,枚举类型是从java5开始增加的一种引用数据类型
  • 枚举值就是当前类的类型,也就是指向本类的对象,默认使用public static final关键字共同像是,因此采取枚举类型.的方式调用
  • 枚举类中可以自定义构造方法,但是构造方法的修饰符必须是private,默认也是私有的

DirectionEnum.java

package com.lagou.task10;

/**
 * 编程实现所有方向的枚举,所有的方向:向上、向下、向左、向右  枚举类型要求所有枚举值必须放在枚举类型的最前面
 */
public enum  DirectionEnum {

    // 2.声明本类类型的引用指向本类类型的对象
    UP("向上"),DOWN("向下"),LEFT("向左"),RIGHT("向右");
    private final String desc; // 用于描述方向字符串的成员变量

    // 通过构造方法实现成员变量的初始化,更加灵活
    // 1.私有化构造方法,此时该构造方法只能在本类的内部使用
    private DirectionEnum(String desc){
        this.desc = desc;
    }

    // 通过公有的get方法可以在本类的外部访问该类的成员变量的数值
    public String getDesc() {
        return desc;
    }
}

DirectionTest.java

        System.out.println("-----------------------------");
        // 使用一下java5开始的枚举类型
        DirectionEnum de = DirectionEnum.DOWN;
        System.out.println("获取到的方向是:" + de.getDesc()); // 向下

在这里插入图片描述

自定义类和枚举类型在switch结构的使用

DirectionUseTest.java

package com.lagou.task10;

public class DirectionUseTest {

    // 自定义静态方法实现根据参数指定的字符串内容来打印具体的方向信息
    public static void test1(String str){
        switch (str){
            case "向上":
                System.out.println("抬头望明月!");break;
            case "向下":
                System.out.println("低头思故乡!");break;
            case "向左":
                System.out.println("左牵黄"); break;
            case "向右":
                System.out.println("右擎苍");break;
            default:
                System.out.println("没有这样的方向哦!");
        }
    }

    // 自定义静态方法实现根据参数指定的枚举类型具体的方向信息
    public static void test2(DirectionEnum de){
        switch (de){
            case UP:
                System.out.println("抬头望明月!");break;
            case DOWN:
                System.out.println("低头思故乡!");break;
            case LEFT:
                System.out.println("左牵黄"); break;
            case RIGHT:
                System.out.println("右擎苍");break;
            default:
                System.out.println("没有这样的方向哦!");
        }
    }
    public static void main(String[] args) {

        DirectionUseTest.test1(Direction.UP.getDesc());
        DirectionUseTest.test1("今天是个好日子!");

        System.out.println("-----------------------------");
        DirectionUseTest.test2(DirectionEnum.DOWN);
        //DirectionUseTest.test2("今天是个好日子!"); Error:类型不匹配 减少来出错的可能性

    }
}
Enum类的概念和常用方法
  • 所有的枚举类型都继承自java.lang.Enum类,常用方法如下:

在这里插入图片描述

前三个DirectionEnumTest.java测试

package com.lagou.task10;

/**
 * 编程实现方向枚举类的测试,调用从Enum类中继承下来的方法
 */
public class DirectionEnumTest {

    public static void main(String[] args) {

        // 1.获取DirectionEnum类型中所有的枚举对象
        DirectionEnum[] arr = DirectionEnum.values();
        // 2.打印每个枚举对象在枚举类型中的名称和和索引位置
        for (int i = 0; i < arr.length; i++){
            System.out.println("获取到的枚举对象名称是:" + arr[i].toString());
            System.out.println("获取到的枚举对象对应的索引位置:" + arr[i].ordinal()); // 和数组一样下标从0开始
            System.out.println("");
        }
    }
}

在这里插入图片描述

package com.lagou.task10;

/**
 * 编程实现方向枚举类的测试,调用从Enum类中继承下来的方法
 */
public class DirectionEnumTest {

    public static void main(String[] args) {

        // 1.获取DirectionEnum类型中所有的枚举对象
        DirectionEnum[] arr = DirectionEnum.values();
        // 2.打印每个枚举对象在枚举类型中的名称和和索引位置
        for (int i = 0; i < arr.length; i++){
            System.out.println("获取到的枚举对象名称是:" + arr[i].toString());
            System.out.println("获取到的枚举对象对应的索引位置:" + arr[i].ordinal()); // 和数组一样下标从0开始
            System.out.println("");
        }
        System.out.println("-------------------------------");
        // 3.根据参数指定的字符串得到枚举类型的对象,也就是将字符串转换为对象
        //DirectionEnum de = DirectionEnum.valueOf("向上");  // 编译ok 运行发生IllegalArgumentException非法参数异常,算数异常、数组越界异常、空指针异常、类型转换异常
        //DirectionEnum de = DirectionEnum.valueOf("UP LEFT"); 要求字符串名称必须在枚举对象中存在
        DirectionEnum de = DirectionEnum.valueOf("DOWN");
        System.out.println("转换出来的枚举对象名称是: " + de.toString());
        System.out.println("转换出来的枚举对象名称是: " + de); // 当打印引用变量时,会自动调用toString方法

        System.out.println("-------------------------------");
        // 4.使用获取到的枚举对象与枚举类中已有的对象比较先后顺序
        for(int i = 0; i < arr.length; i++){
            // 当调用对象在参数对象之后时,获取到的比较结果为 正数
            // 当调用对象在参数对象相同时,获取到的比较结果为 零
            // 当调用对象在参数对象之前时,获取到的比较结果为 负数
            System.out.println("调用对象与数组中对象比较的先后顺序结果是:" + de.compareTo(arr[i]));
        }
    }
}

在这里插入图片描述

枚举类实现接口的方式
  • 枚举类实现接口后需要重写抽象方法,而重写方法的方式有两种:重写一个,或者每个对象都重写

DirectionEnum.java

package com.lagou.task10;

/**
 * 编程实现所有方向的枚举,所有的方向:向上、向下、向左、向右  枚举类型要求所有枚举值必须放在枚举类型的最前面
 */
public enum  DirectionEnum implements DirectionInterface{

    // 2.声明本类类型的引用指向本类类型的对象
    // 匿名内部类的语法格式:接口/父类类型 引用变量名 = new 接口/父类类型(){方法的重写}
    // public static final Direction UP = new Direction("向上"){};
    UP("向上"){
        @Override
        public void show() {
            System.out.println("贪吃蛇向上移动了一下!");
        }
    },DOWN("向下") {
        @Override
        public void show() {
            System.out.println("贪吃蛇向下移动了一下!");
        }
    },LEFT("向左") {
        @Override
        public void show() {
            System.out.println("左移了一下!");
        }
    },RIGHT("向右") {
        @Override
        public void show() {
            System.out.println("右移了一下!");
        }
    };
    private final String desc; // 用于描述方向字符串的成员变量

    // 通过构造方法实现成员变量的初始化,更加灵活
    // 1.私有化构造方法,此时该构造方法只能在本类的内部使用
    private DirectionEnum(String desc){
        this.desc = desc;
    }

    // 通过公有的get方法可以在本类的外部访问该类的成员变量的数值
    public String getDesc() {
        return desc;
    }

    // 整个枚举类型只重写一次,所有对象调用同一个
    /*@Override
    public void show() {
        System.out.println("现在可以实现接口中抽象方法的重写了!");
    }*/

}

在这里插入图片描述

DirectionEnumTest.java

package com.lagou.task10;

/**
 * 编程实现方向枚举类的测试,调用从Enum类中继承下来的方法
 */
public class DirectionEnumTest {

    public static void main(String[] args) {

        // 1.获取DirectionEnum类型中所有的枚举对象
        DirectionEnum[] arr = DirectionEnum.values();
        // 2.打印每个枚举对象在枚举类型中的名称和和索引位置
        for (int i = 0; i < arr.length; i++){
            System.out.println("获取到的枚举对象名称是:" + arr[i].toString());
            System.out.println("获取到的枚举对象对应的索引位置:" + arr[i].ordinal()); // 和数组一样下标从0开始
            System.out.println("");
        }
        System.out.println("-------------------------------");
        // 3.根据参数指定的字符串得到枚举类型的对象,也就是将字符串转换为对象
        //DirectionEnum de = DirectionEnum.valueOf("向上");  // 编译ok 运行发生IllegalArgumentException非法参数异常,算数异常、数组越界异常、空指针异常、类型转换异常
        //DirectionEnum de = DirectionEnum.valueOf("UP LEFT"); 要求字符串名称必须在枚举对象中存在
        DirectionEnum de = DirectionEnum.valueOf("DOWN");
        System.out.println("转换出来的枚举对象名称是: " + de.toString());
        System.out.println("转换出来的枚举对象名称是: " + de); // 当打印引用变量时,会自动调用toString方法

        System.out.println("-------------------------------");
        // 4.使用获取到的枚举对象与枚举类中已有的对象比较先后顺序
        for(int i = 0; i < arr.length; i++){
            // 当调用对象在参数对象之后时,获取到的比较结果为 正数
            // 当调用对象在参数对象相同时,获取到的比较结果为 零
            // 当调用对象在参数对象之前时,获取到的比较结果为 负数
            System.out.println("调用对象与数组中对象比较的先后顺序结果是:" + de.compareTo(arr[i]));
        }
        System.out.println("-------------------------------");
        // 5.使用数组中每个DirectionEnum对象都去调用show方法测试
        for (int i = 0; i < arr.length; i++){
            arr[i].show();
        }
    }
}

在这里插入图片描述

注解的基本概念
  • 注解(Annotation)又叫标注,是从java5开始增加的一种引用数据类型
  • 注解本质上就是代码中的特殊标记,通过这些标记可以在编译、类加载、以及运行时执行指定的处理
注解的定义和使用
注解的语法格式
  • 访问修饰符 @interface 注解名称{

    ​ 注解成员;

    }

  • 自定义注解自动继承java.lang.annotation.Annotation接口

  • 通过@注解名称的方式可以修饰包、类、成员方法、成员变量、构造方法、参数、局部变量的声明

注解的使用方式
  • 注解体中只有成员变量没有成员方法,而注解的成员变量以“无形参的方法”形式来声明,其方法名定义来该成员变量的名字,其返回值定义来该成员变量的类型
  • 如果注解只有一个参数成员,建议使用参数名为value,而类型只能是八种基本数据类型、String类型、Class类型、enum类型及Annotation类型

MyAnnotation.java

package com.lagou.task10;

// 若一个注解中没有任何的成员,则这样的注解叫做标记注解/标示注解
public @interface MyAnnotation {

    //public Direction value(); // 声明一个String类型的成员变量,名字为value  类型有要求
    public String value() default "123"; // 声明一个String类型的成员变量,名字为value
    public String value2();
}

Person.java

package com.lagou.task10;
// 表示将标签MyAnnotation帖在Person类的代码中,使用注解时采用 成员参数名 = 成员参数值,...
//@MyAnnotation(value = "hello",value2 = "world")
@MyAnnotation(value2 = "world")
public class Person {
    private String name;
    private int age;
}
元注解的概念和@Retentio的使用
元注解的概念
  • 元注解是可以注解到注解上的注解,或者说元注解是一种基本注解,但是它能够应用到其它的注解上面
  • 元注解主要有@Retention、@Documented、@Target、@Inherited、@Repeatable
元注解@Retention
  • @Retention应用到一个注解上用于说明该注解的生命周期,取值如下:
  • RetentionPolicy.SOURCE 注解只在源码阶段保留,在编译器进行编译时它将被丢弃
  • RetentionPolicy.CLASS 注解只被保留到编译进行的时候,它并不会被加载到JVM中,默认方式
  • RetentionPolicy.RUNTIME 注解可以保留到程序运行的时候,它会被加载进入到JVM中,所以在程序运行时可以获取到它们
package com.lagou.task10;

import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;

//@Retention(RetentionPolicy.SOURCE)   // 表示下面的注解在源代码中有效
//@Retention(RetentionPolicy.CLASS)    // 表示下面的注解在字节码文件中有效,默认方式
@Retention(RetentionPolicy.RUNTIME)  // 表示下面的注解在运行时有效
// 若一个注解中没有任何的成员,则这样的注解叫做标记注解/标示注解
public @interface MyAnnotation {

    //public Direction value(); // 声明一个String类型的成员变量,名字为value  类型有要求
    public String value() default "123"; // 声明一个String类型的成员变量,名字为value
    public String value2();
}
@Documented的使用
元注解@Documented
  • 使用javadoc工具可以从程序源代码中抽取类、方法、成员等注释形成一个和源代码配套的API帮助文档,而该工具抽取时默认不包括注解内容

  • @Documented用于指定被该注解将被javadoc工具提取成文档

  • 定义@Documented的注解必须设置Retention值为RUNTIME

MyAnnotation.java

package com.lagou.task10;

import java.lang.annotation.Documented;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;

//@Retention(RetentionPolicy.SOURCE)   // 表示下面的注解在源代码中有效
//@Retention(RetentionPolicy.CLASS)    // 表示下面的注解在字节码文件中有效,默认方式
@Retention(RetentionPolicy.RUNTIME)    // 表示下面的注解在运行时有效
@Documented                            // 表示下面的注解信息可以被javadoc工具提取到API文档中,很少使用
// 若一个注解中没有任何的成员,则这样的注解叫做标记注解/标示注解
public @interface MyAnnotation {

    //public Direction value(); // 声明一个String类型的成员变量,名字为value  类型有要求
    public String value() default "123"; // 声明一个String类型的成员变量,名字为value
    public String value2();
}
@Target和@Inherited的使用
元注解@Target
  • @Target用于指定被修饰的注解能用于哪些元素的修饰,取值如下:

在这里插入图片描述
在这里插入图片描述

MyAnnotation.java

package com.lagou.task10;

import java.lang.annotation.*;

//@Retention(RetentionPolicy.SOURCE)   // 表示下面的注解在源代码中有效
//@Retention(RetentionPolicy.CLASS)    // 表示下面的注解在字节码文件中有效,默认方式
@Retention(RetentionPolicy.RUNTIME)    // 表示下面的注解在运行时有效
@Documented                            // 表示下面的注解信息可以被javadoc工具提取到API文档中,很少使用
// 表示下面的注解可以用于类型的修饰、构造方法、成员变量、成员方法、参数的修饰
@Target({/*ElementType.TYPE,*/ElementType.CONSTRUCTOR,ElementType.FIELD,ElementType.METHOD,ElementType.PARAMETER})

// 若一个注解中没有任何的成员,则这样的注解叫做标记注解/标示注解
public @interface MyAnnotation {

    //public Direction value(); // 声明一个String类型的成员变量,名字为value  类型有要求
    public String value() default "123"; // 声明一个String类型的成员变量,名字为value
    public String value2();
}

Person.java

package com.lagou.task10;
// 表示将标签MyAnnotation帖在Person类的代码中,使用注解时采用 成员参数名 = 成员参数值,...
//@MyAnnotation(value = "hello",value2 = "world")
//@MyAnnotation(value2 = "world")
public class Person {
    /**
     * name是用于描述姓名的成员变量
     */
    @MyAnnotation(value2 = "1")
    private String name;
    /**
     * age是用于描述年龄的成员变量
     */
    private int age;

    /**
     * 自定义成员方法实现无参构造方法
     */
    @MyAnnotation(value2 = "2")
    public Person() {
    }

    /**
     * 编程实现有参构造方法
     * @param name
     * @param age
     */
    public Person(@MyAnnotation(value2 = "4") String name, int age) {
        this.name = name;
        this.age = age;
    }

    /**
     *自定义成员方法实现特征的获取和修改
     * @return
     */
    @MyAnnotation(value2 = "3")
    public String getName() { return name; }

    public void setName(String name) { this.name = name; }

    public int getAge() { return age; }

    public void setAge(int age) { this.age = age; }
}
元注解@Inherited
  • @Inherited并不是说注解本身可以继承,而是说明如果一个超类被该注解标记过的注解进行注解时,如果子类没有被任何注解应用时,则子类就继承超类的注解

在这里插入图片描述

@Repeatable的使用
元注解@Repeatable
  • @Repeatable表示自然可重复的含义,从java8开始增加的新特性
  • 从Java8开始对元注解@Target的参数类型ElementType枚举值增加了两个:
  • 其中ElementType.TYPE_PARAMETER表示该注解能写在类型变量的声明语句中,如:泛型
  • 其中ElementType.TYPE_USE表示该注解能写在使用类型的任何语句中

注解:ManType.java

package com.lagou.task10;

import java.lang.annotation.ElementType;
import java.lang.annotation.Repeatable;
import java.lang.annotation.Target;

/**
 * 自定义注解用于描述人物的角色
 */
@Repeatable(value = ManTypes.class)
@Target(ElementType.TYPE_USE)
public @interface ManType {
    String value() default "";
}

类:Man.java

package com.lagou.task10;

@ManType(value = "职工")
@ManType(value = "超人")
//@ManTypes({@ManType(value = "职工"),@ManType(value = "超人")})  //在java8以前处理多个注解的方式

public class Man {

    public static void main(String[] args) {
        int ia = 97;
        char c1 = (@ManType char) ia;
    }
}

注解:ManTypes.java

package com.lagou.task10;

import java.lang.annotation.ElementType;
import java.lang.annotation.Target;

/**
 * 自定义注解用于描述多种角色
 */
@Target(ElementType.TYPE_USE)
public @interface ManTypes {
    ManType[] value();
}
常见的预制注解
  • 预制注解就是java语言自身提供的注解,具体如下:‘
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
总结

1、内部类

概念、普通内部类、静态内部类、局部内部类、匿名内部类、回调模式

2、枚举类型

今天概念、自定义枚举、enum关键字、继承自Enum类、实现接口等

3、注解

概念、自定义注解、使用、元注解、预制注解等

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值