今日学习内容

  • Junit单元测试
  • 反射
  • 注解

1 Junit单元测试

  • @Test注解

    • 它不是JDK提供的注解,是Junit提供的,所以要将Junit的依赖先导入到模块中才可以使用
    • @Test注解是加在方法之上的
1.1 测试分类
  • 黑盒测试:不需要写代码,给输入值,看程序是否能够输出期望的值。
  • 白盒测试:需要写代码的。关注程序具体的执行流程。
1.2 Junit使用:白盒测试
* 步骤:
    1. 定义一个测试类(测试用例)
        * 建议:
            * 测试类名:被测试的类名Test CalculatorTest
            * 包名:xxx.xxx.xx.test cn.noisky.test

    2. 定义测试方法:可以独立运行
        * 建议:
            * 方法名:test测试的方法名 testAdd()  
            * 返回值:void
            * 参数列表:空参

    3. 给方法加@Test
    4. 导入junit依赖环境

* 判定结果:
    * 红色:失败
    * 绿色:成功
    * 一般我们会使用断言操作来处理结果
        * Assert.assertEquals(期望的结果,运算的结果);

* 补充:
    * @Before:
        * 修饰的方法会在测试方法之前被自动执行
    * @After:
        * 修饰的方法会在测试方法执行之后自动被执行

2 反射

  • 框架:半成品软件。可以在框架的基础上进行软件开发,简化编码
  • 反射:将类的各个组成部分封装为其他对象,在运行阶段可以获取和操作这些对象,这就是反射机制
  • 好处:

    1. 可以在程序运行过程中,操作这些对象。
    2. 可以解耦,提高程序的可扩展性。

2.1 获取Class对象的方式:

1. Class.forName("全类名"):将字节码文件加载进内存,返回Class对象
    * 多用于配置文件,将类名定义在配置文件中。读取文件,加载类
2. 类名.class:通过类名的属性class获取
    * 多用于参数的传递
3. 对象.getClass():getClass()方法在Object类中定义着。
    * 多用于对象的获取字节码的方式

* 结论:
    同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。

2.2 Field(成员变量)

1.如何获取

  • Class类的方法

    方法说明
    Field[] getFields()获取所有public修饰的成员变量
    Field getField(String name)获取指定名称的 public修饰的成员变量
    Field[] getDeclaredFields()获取所有的成员变量,不考虑修饰符
    Field getDeclaredField(String name)获取指定名称的成员变量,不考虑修饰符

2.如何操作

  • Field类的方法

    • Object get(Object obj)
    • void set(Object obj, Object value)
    • void setAccessible(boolean flag) :flag设置为true,就是暴力反射
  • 取值(get)
  • 赋值(set)
  • 步骤:

    • public修饰的成员变量

        1、获取字节码对象
        2、获取指定名称的成员变量的Field对象
        3、创建对象
        4、调用get或者set取值和赋值
      
        // 1、
        Class personClass = Person.class;
        // 2、
        Field a = personClass.getField("a");// public String a;
        // 3、
        Person p = new Person();
        // 4、
        // 获取
        Object value = a.get(p);// Object value = p.a;
        System.out.println(value);
        // 赋值
        a.set(p,"张三");        // p.a = "张三";
        System.out.println(p);
    • 非public修饰的成员变量

      1、获取字节码对象
      2、获取指定名称的成员变量的Field对象
      3、创建对象
      4、暴力反射
      5、调用get或者set取值和赋值

2.3 Constructor(构造方法)

一、如何获取

  • Class类中的方法

    方法说明
    Constructor<?>[] getConstructors()返回一个包含 Constructor对象的数组, Constructor对象反映了由该 Class对象表示的类的所有公共构造函数。
    Constructor getConstructor(Class<?>... parameterTypes)返回一个 Constructor对象,该对象反映由该 Class对象表示的类的指定公共构造函数。
    Constructor<?>[] getDeclaredConstructors()返回反映由该 Class对象表示的类声明的所有构造函数的 Constructor对象的数组。
    Constructor getDeclaredConstructor(Class<?>... parameterTypes)返回一个 Constructor对象,该对象反映由此 Class对象表示的类或接口的指定构造函数。
    T newInstance()使用无参构造创建对象

二、如何操作(创建对象)

  • Constructor类中的方法

    • T newInstance(Object... initargs)
  • 步骤:

    • 使用有参构造创建对象

      (1)获取字节码对象

      (2)获取指定的构造方法的Constructor对象

      (3)调用newInstance(T...)方法执行构造方法创建对象

      // (1)
      Class personClass = Person.class;
      // (2)
      //public Person(String name, int age) {}
      Constructor constructor = personClass.getConstructor(String.class, int.class);
      // (3)
      // Person person = new Person("张三",23);
      Object person = constructor.newInstance("张三", 23);
  • 使用无参构造创建对象

    • 无参构造方式一:

      (1)获取字节码对象
      (2)获取指定的构造方法的Constructor对象
      (3)调用newInstance(T...)方法执行构造方法创建对象

      // (1)
      Class personClass = Person.class;
      // (2)
      //public Person() {}
      Constructor constructor = personClass.getConstructor();
      // (3)
      // Person person = new Person();
      Object person = constructor.newInstance();
    • 无参构造方式二:
      (1)获取字节码对象
      (2)使用字节码对象调用newInstance()方法创建对象

      Class personClass = Person.class;
      Object person = personClass.newInstance();

2.4 Method(成员方法)

一、如何获取

  • Class类中的方法

    方法说明
    Method[] getMethods()返回一个包含 方法对象的数组, 方法对象反映由该 Class对象表示的类或接口的所有公共方法,包括由类或接口声明的对象以及从超类和超级接口继承的类。
    Method getMethod(String name, Class<?>... parameterTypes)返回一个 方法对象,该对象反映由该 Class对象表示的类或接口的指定公共成员方法。
    Method[] getDeclaredMethods()返回一个包含 方法对象的数组, 方法对象反映由 Class对象表示的类或接口的所有声明方法,包括public,protected,default(package)访问和私有方法,但不包括继承方法。
    Method getDeclaredMethod(String name, Class<?>... parameterTypes)返回一个 方法对象,它反映此表示的类或接口的指定声明的方法 Class对象。

二、如何操作(调用)

Method类中的方法

  • Object invoke(Object obj, Object... args)
    步骤:
(1)获取字节码对象
(2)获取指定方法名称和参数类型的Method对象
(3)创建对象
(4)使用Method对象调用invoke方法
        
Class personClass = Person.class;
Method eat_method = personClass.getMethod("eat");// eat()
Person p = new Person();
eat_method.invoke(p);// p.eat();


Class personClass = Person.class;
Method eat_method = personClass.getMethod("eat",String.class);// eat(String s)
Person p = new Person();
eat_method.invoke(p,"饭");// p.eat("饭");
  • 获取全类名

    • String getName()

3 注解

3.1 概念

  • 概念:说明程序的。给计算机看的
  • 注释:用文字描述程序的。给程序员看的
  • 定义:注解(Annotation),也叫元数据。一种代码级别的说明。它是JDK1.5及以后版本引入的一个特性,与类、接口、枚举是在同一个层次。它可以声明在包、类、字段、方法、局部变量、方法参数等的前面,用来对这些元素进行说明,注释。
  • 概念描述:

    • JDK1.5之后的新特性
    • 说明程序的
    • 使用注解:@注解名称

3.2 作用分类

  • 作用分类:
    ①编写文档:通过代码里标识的注解生成文档【生成文档doc文档】
    ②代码分析:通过代码里标识的注解对代码进行分析【使用反射】
    ③编译检查:通过代码里标识的注解让编译器能够实现基本的编译检查【Override】

3.3 JDK中预定义的一些注解

  • @Override :检测被该注解标注的方法是否是继承自父类(接口)的
  • @Deprecated:该注解标注的内容,表示已过时
  • @SuppressWarnings:压制警告

    • 一般传递参数all @SuppressWarnings("all")

3.4 自定义注解

  • 自定义注解
  • 格式:

    元注解
    public @interface 注解名称{
    属性列表;
    }

  • 本质:注解本质上就是一个接口,该接口默认继承Annotation接口
  • public interface MyAnno extends java.lang.annotation.Annotation {}
  • 属性:接口中的抽象方法
  • 要求:

    1. 属性的返回值类型有下列取值

      • 基本数据类型
      • String
      • 枚举
      • 注解
      • 以上类型的数组
    2. 定义了属性,在使用时需要给属性赋值

      1. 如果定义属性时,使用default关键字给属性默认初始化值,则使用注解时,可以不进行属性的赋值。
      2. 如果只有一个属性需要赋值,并且属性的名称是value,则value可以省略,直接定义值即可。
      3. 数组赋值时,值使用{}包裹。如果数组中只有一个值,则{}可以省略
  • 元注解:用于描述注解的注解

    • @Target:描述注解能够作用的位置

      • ElementType取值:

        • TYPE:可以作用于类上
        • METHOD:可以作用于方法上
        • FIELD:可以作用于成员变量上
    • @Retention:描述注解被保留的阶段

      • @Retention(RetentionPolicy.RUNTIME):当前被描述的注解,会保留到class字节码文件中,并被JVM读取到
    • @Documented:描述注解是否被抽取到api文档中
    • @Inherited:描述注解是否被子类继承

3.5 注解中定义属性

在程序使用(解析)注解:获取注解中定义的属性值

  1. 获取注解定义的位置的对象 (Class,Method,Field)
  2. 获取指定的注解

    • getAnnotation(Class)
//其实就是在内存中生成了一个该注解接口的子类实现对象

        public class ProImpl implements Pro{
            public String className(){
                return "cn.noisky.annotation.Demo1";
            }
            public String methodName(){
                return "show";
            }
        }
  1. 调用注解中的抽象方法获取配置的属性值

4 小结

  1. 以后大多数时候,我们会使用注解,而不是自定义注解
  2. 注解给谁用?

    1. 编译器
    2. 给解析程序用
  3. 注解不是程序的一部分,可以理解为注解就是一个标签