03 - Java核心类库 — 枚举、注解与反射

目前阶段这块知识点了解即可,但在学习框架非常重要,能更好的理解框架。枚举能更好的管理一组相同的常量,内省是基于反射的基础,按照Java bean的标准延伸出来一套操作的API。

反射和内省,如果不进行框架开发,则很少使用;注解一般使用定义好的,自定义注解用的比较少。

一、枚举

1. 简介

JDK1.5引入了新的类型——枚举。

在JDK1.5 之前,我们定义常量都是: public static fianl… 。很难管理。

枚举,可以把相关的常量分组到一个枚举类型里,而且枚举提供了比常量更多的方法。

==用于定义有限数量的一组同类常量 ==,例如:

  • 错误级别: 低、中、高
  • 一年的四季: 春、夏、秋、冬
  • 商品的类型: 美妆、手机、电脑、男装、女装…

在枚举类型中定义的常量是该枚举类型的实例
在这里插入图片描述

2. 定义格式

2.1 Java1.5版本之前

Level_1.java:

/**
 * @Author: 爱摸鱼的TT~
 * @Description:枚举 JDK1.5之前的写法
 * @Date Created in 2021-03-21 9:28
 * @Modified By:
 */
public class Level_1 {
    // 定义枚举常量:private static final...
    public static final Level_1 LOW = new Level_1(1);
    public static final Level_1 MEDIUM = new Level_1(50);
    public static final Level_1 HIGH = new Level_1(100);
    // 定义一个级别变量
    private int levelValue;
    // 级别类的构造方法
    // 注意:这里只能用私有修饰,因为枚举类型是不能被外部直接调用的,所以要私有化。
    private Level_1(int levelValue){
        this.levelValue = levelValue;
    }
    // getter和setter方法
    public int getLevelValue() {
        return levelValue;
    }

    //不要为枚举类中的属性提供set方法,不符合枚举最初设计初衷。
    //初衷:一旦定义了枚举,最好不要妄图修改里面的值,除非修改是必要的。
    /*public void setLevelValue(int levelValue) {
        this.levelValue = levelValue;
    }*/
}

Demo.java:

/**
 * @Author: 爱摸鱼的TT~
 * @Description: 测试类
 * @Date Created in 2021-03-21 10:04
 * @Modified By:
 */
public class Demo {
    public static void main(String[] args) {
        System.out.println(Level_1.MEDIUM.getLevelValue());//JDK1.5前
    }
}

2.2 Java1.5版本之后

Level_2.java

/**
 * @Author: 爱摸鱼的TT~
 * @Description:枚举 JDK1.5后的写法
 * @Date Created in 2021-03-21 9:36
 * @Modified By:
 */
// 注意这里将class换成enum
public enum Level_2 {
    // 枚举写法,这里其实就是调用此构造方法。
    LOW(1),MEDIUM(50),HIGH(100);
    // 也可以这样来表示,直接从顺序上判断优先级
    //LOW,MEDIUM,HIGH;

    private int levelValue;

    // private修饰的构造方法 表示外界不能通过构造方法实例化对象 同时也不能添加类型
    private Level_2(int levelValue){
        this.levelValue = levelValue;
    }

    public int getLevelValue() {
        return levelValue;
    }

    public void setLevelValue(int levelValue) {
        this.levelValue = levelValue;
    }
}

Demo.java

public class Demo {
    public static void main(String[] args) {
        System.out.println(Level_2.LOW.getLevelValue());
    }
}

3. 枚举类的主要方法

3.1 归纳

在这里插入图片描述

3.2 实例

Level_2.java 【上面已有对应的代码】

Demo.java

/**
 * @Author: 爱摸鱼的TT~
 * @Description: 测试类
 * @Date Created in 2021-03-21 10:04
 * @Modified By:
 */
public class Demo {
    public static void main(String[] args) {
        System.out.println(Level_1.MEDIUM.getLevelValue());//JDK1.5前
        // compareTo
        System.out.println(Level_2.LOW.compareTo(Level_2.MEDIUM));// -1
        System.out.println(Level_2.LOW.compareTo(Level_2.HIGH));// -2
        System.out.println(Level_2.MEDIUM.compareTo(Level_2.LOW));// 1
        System.out.println(Level_3.MEDIUM.compareTo(Level_3.HIGH));// -1
        // name
        System.out.println(Level_3.LOW.name());// LOW
        System.out.println(Level_3.MEDIUM.name());// MEDIUM
        System.out.println(Level_2.HIGH.name());// HIGH
        // ordinal
        System.out.println(Level_3.LOW.ordinal());// 0
        System.out.println(Level_3.MEDIUM.ordinal());// 1
        System.out.println(Level_2.HIGH.ordinal());// 2
        //toString
        System.out.println(Level_2.LOW.toString());// LOW
        System.out.println(Level_3.MEDIUM.toString());// MEDIUM
        System.out.println(Level_3.HIGH.toString());// HIGH
        // valueOf
        Level_3 s = Enum.valueOf(Level_3.class, "LOW");
        System.out.println(s.name());// LOW
        //equals:通常使用switch语句和equal语句。
        System.out.println("------------------------------------------");
        Level_3.LOW.show();// 低级别
        Level_3.MEDIUM.show();// 中级别
        Level_3.HIGH.show();// 高级别
    }
}

4. 实现接口的枚举类

所有的枚举都继承自java.lang.Enum类。由于Java 不支持多继承,所以枚举对象不能再继承其他类。

每个枚举对象,都可以实现自己的抽象方法

实例

Level.java
在这里插入图片描述
main函数
在这里插入图片描述
存在的问题:LOW、MEDIUM、HIGH都是一样的输出,为了解决其各自输出,如下代码块:

/**
 * @Author: 爱摸鱼的TT~
 * @Description: 枚举 最简约写法,以及实现Level接口
 * @Date Created in 2021-03-21 9:39
 * @Modified By:
 */
public enum Level_3 implements Level{
    // 最简约的写法,无给其赋具体的值,就单纯定义了级别类型
    // LOW,MEDIUL,HIGH;

    // 每个枚举常量实现Level接口,就能输出不一样的值。
    // 每个枚举对象,都可以实现自己的抽象方法
    LOW{
        @Override
        public void show() {
            System.out.println("低级别");
        }
    },MEDIUM(){// 括号表示调用无参构造方法
        @Override
        public void show() {
            System.out.println("中级别");
        }
    },HIGH{
        @Override
        public void show() {
            System.out.println("高级别");
        }
    };
}
interface Level{
    void show();
}

5. 枚举类的注意事项

  • 一旦定义了枚举,最好不要妄图修改里面的值,除非修改是必要的;
  • 枚举类默认继承的是java.lang.Enum类而不是Object类;
  • 枚举类不能有子类,因为其枚举类默认被final修饰;
  • 只能有private构造方法;
  • switch中使用枚举时,直接使用常量名,不用携带类名;
  • 不能定义name属性,因为自带name属性;
  • 不要为枚举类中的属性提供set方法,不符合枚举最初设计初衷;

二、注解

1. 简介

注解与注释:注释是为了程序员更好的理解代码,只能存在于源文件中,一旦编译变成.class文件,便会忽略掉注释。注解可以理解为让机器理解的,可以保留到代码执行的阶段。

Java 注解(Annotation)又称 Java 标注,是 JDK5.0 引入的一种注释机制。

Java 语言中的类、方法、变量、参数和包等都可以被标注。和注释不同,Java 标注可以通过反射获取标注内容。在编译器生成类文件时,标注可以被嵌入到字节码中。Java 虚拟机可以保留标注内容,在运行时可以获取到标注内容 。 当然它也支持自定义 Java 标注。

主要用于:

  • 编译格式检查
  • 反射中解析
  • 生成帮助文档
  • 跟踪代码依赖

2. 学习的重点

理解 Annotation 的关键,是理解 Annotation 的语法和用法.

学习步骤:

  1. 概念
  2. 怎么使用内置注解
  3. 怎么自定义注解
  4. 反射中怎么获取注解内容

3. 内置注解

3.1 概述

@Override : 重写 *****

定义在java.lang.Override

@Deprecated:废弃 *****

定义在java.lang.Deprecated

@SafeVarargs

Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。

@FunctionalInterface: 函数式接口 *****

Java 8 开始支持,标识一个匿名函数或函数式接口。

@Repeatable:标识某注解可以在同一个声明上使用多次

Java 8 开始支持,标识某注解可以在同一个声明上使用多次。

SuppressWarnings:抑制编译时的警告信息。 *****
定义在java.lang.SuppressWarnings
三种使用方式

  • @SuppressWarnings(“unchecked”) [^ 抑制单类型的警告]
  • @SuppressWarnings(“unchecked”,“rawtypes”) [^ 抑制多类型的警告]
  • @SuppressWarnings(“all”) [^ 抑制所有类型的警告]

参数列表

  • all 抑制所有警告
  • boxing 抑制装箱、拆箱操作时候的警告
  • cast 抑制映射相关的警告
  • dep-ann 抑制启用注释的警告
  • deprecation 抑制过期方法警告
  • fallthrough 抑制确在switch中缺失breaks的警告
  • finally 抑制finally模块没有返回的警告
  • hiding 抑制相对于隐藏变量的局部变量的警告
  • incomplete-switch 忽略没有完整的switch语句
  • nls 忽略非nls格式的字符
  • null 忽略对null的操作
  • rawtypes 使用generics时忽略没有指定相应的类型
  • restriction 抑制禁止使用劝阻或禁止引用的警告
  • serial 忽略在serializable类中没有声明serialVersionUID变量
  • static-access 抑制不正确的静态访问方式警告
  • synthetic-access 抑制子类没有按最优方法访问内部类的警告
  • unchecked 抑制没有进行类型检查操作的警告
  • unqualified-field-access 抑制没有权限访问的域的警告
  • unused 抑制没被使用过的代码的警告

3.2 理解

1)Override重写

IDEA中每次保存代码,都会将.java文件转换为.class文件, 这时会检查属性或方法的注解,查看其是否符合注解的格式要求,若不符合则给出错误信息:
在这里插入图片描述

2)Deprecated废弃

对于前期编写的一些方法,后期可能不再需要,若此时将其删除,则会使之前调用此方法的地方出现问题,所以建议扩展代码,而不是删除。

public class Demo {
    public static void main(String[] args) {
        Person p = new Person();
        p.setAge(12);
 
    }
}
class Person{
    private int age;
 
    public int getAge() {
        return age;
    }
 
    /**
     * 此方法已经被废弃,请通过setAge2进行操作
     * @param age
     */
    @Deprecated
    public void setAge(int age) {
        this.age = age;
    }
 
    public void setAge2(int age) {
        if(age < 0 || age > 120) {
            throw new RuntimeException("该年龄不合理");
        }
        this.age = age;
    }
}

在这里插入图片描述在这里插入图片描述
3)SuppressWarnings:抑制编译时的警告信息
在这里插入图片描述在这里插入图片描述

4. 元注解

4.1 简介

作用在其他注解的注解

4.2 有哪些?

@Retention - 标识这个注解怎么保存,是只在代码中,还是编入class文件中,或者是在运行时可以通过反射访问。

@Documented - 标记这些注解是否包含在用户文档中 javadoc。

@Target - 标记这个注解应该是哪种 Java 成员。

@Inherited - 标记这个注解是自动继承的

  • 子类会继承父类使用的注解中被@Inherited修饰的注解
  • 接口继承关系中,子接口不会继承父接口中的任何注解,不管父接口中使用的注解有没有 被@Inherited修饰
  • 类实现接口时不会继承任何接口中定义的注解

5. 自定义注解

在这里插入图片描述

5.1 注解架构

  1. Annotation与RetentionPolicy 与ElementType

每 1 个 Annotation 对象,都会有唯一的 RetentionPolicy 属性;至于 ElementType 属性,则有 1~n个。

(02) ElementType(注解的用途类型)

“每 1 个 Annotation” 都与 “1~n 个 ElementType” 关联。当 Annotation 与某个 ElementType 关联时,就意味着:Annotation有了某种用途。例如,若一个 Annotation 对象是 METHOD 类型,则该Annotation 只能用来修饰方法。

package java.lang.annotation; 
public enum ElementType { 
    TYPE, /* 类、接口(包括注释类型)或枚举声明 */ 
    FIELD, /* 字段声明(包括枚举常量) */ 
    METHOD, /* 方法声明 */ 
    PARAMETER, /* 参数声明 */ 
    CONSTRUCTOR, /* 构造方法声明 */ 
    LOCAL_VARIABLE, /* 局部变量声明 */ 
    ANNOTATION_TYPE, /* 注释类型声明 */ 
    PACKAGE /* 包声明 */ 
}

(03) RetentionPolicy(注解作用域策略)
“每 1 个 Annotation” 都与 “1 个 RetentionPolicy” 关联。

  • a) 若 Annotation 的类型为 SOURCE,则意味着:Annotation 仅存在于编译器处理期间,编译器处理完之后,该Annotation 就没用了。 例如," @Override" 标志就是一个Annotation。当它修饰一个方法的时候,就意味着该方法覆盖父类的方法;并且在编译期间会进行语法检查!编译器处理完后,"@Override"
    就没有任何作用了。
  • b) 若 Annotation 的类型为 CLASS,则意味着:编译器将 Annotation 存储于类对应的 .class 文件中,它是Annotation 的默认行为。
  • c) 若 Annotation 的类型为 RUNTIME,则意味着:编译器将 Annotation 存储于 class 文件中,并且可由JVM读入。
package java.lang.annotation; 
public enum RetentionPolicy { 
    SOURCE, /* Annotation信息仅存在于编译器处理期间,编译器处理完之后就没有该 Annotation信息了 */ 
    CLASS, /* 编译器将Annotation存储于类对应的.class文件中。默认行为 */ 
    RUNTIME /* 编译器将Annotation存储于class文件中,并且可由JVM读入 */ 
}

5.2 定义格式

@interface 自定义注解名{}

5.3 注意事项

  1. 定义的注解,自动继承了java.lang,annotation.Annotation接口

  2. 注解中的每一个方法,实际是声明的注解配置参数

  • 方法的名称就是 配置参数的名称
  • 方法的返回值类型,就是配置参数的类型。只能是:基本类型/Class/String/enum
  1. 可以通过default来声明参数的默认值

  2. 如果只有一个参数成员,一般参数名为value

  3. 注解元素必须要有值,我们定义注解元素时,经常使用空字符串、0作为默认值。

5.4 案例

在这里插入图片描述
在这里插入图片描述
注解传参
在这里插入图片描述
也可以不传递全部参数
在这里插入图片描述

三、反射

1. 概述

JAVA反射机制是在运行状态(程序已经执行起来后)中,获取任意一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性 !;

这种在运行状态动态获取信息以及动态调用对象方法的功能被称为java语言的反射机制。
在这里插入图片描述
spring是很早就编写的框架,用它来管理现在创建的对象,所用到的就是反射机制

2. 类加载器

.java文件通过编译得到.class字节码文件,再通过类加载器加载到内存中,才可以运行,根据类创建对象

Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分, 负责动态加载Java类到Java虚拟机的内存空间中。

java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、AppClassLoader。

  • BootstrapClassLoader(引导启动类加载器):嵌在JVM内核中的加载器,该加载器是用C++语言写的,主要负责加载JAVA_HOME/lib下的类库,引导启动类加载器无法被应用程序直接使用。
  • ExtensionClassLoader(扩展类加载器):ExtensionClassLoader是用JAVA编写,且它的父类加载器是Bootstrap。是由sun.misc.Launcher$ExtClassLoader实现的,主要加载JAVA_HOME/lib/ext目录中的类
    库。它的父加载器是BootstrapClassLoader
  • App ClassLoader(应用类加载器):
    AppClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文
    件。它的父加载器为ExtClassLoader
    在这里插入图片描述

类通常是按需加载,即第一次使用该类时才加载。

由于有了类加载器,Java运行时系统不需要知道文件与 文件系统。学习类加载器时,掌握Java的委派概念很重要。

双亲委派模型:如果一个类加载器收到了一个类加载请求,它不会自己去尝试加载这个类,而是把这个请求 转交给父类加载器去完成。每一个层次的类加载器都是如此。因此所有的类加载请求都应该传递到最顶层的 启动类加载器中,只有到父类加载器反馈自己无法完成这个加载请求(在它的搜索范围没有找到这个类) 时,子类加载器才会尝试自己去加载。

委派的好处就是避免有些类被重复加载。
在这里插入图片描述
获得类加载器
在这里插入图片描述
通过类加载器读取配置文件内容
在这里插入图片描述
创建新文件夹resource(截图中应该命名为resource)
在这里插入图片描述
将该文件夹设置为资源根目录
在这里插入图片描述
将config.txt文件放入刚创建的文件夹中,并修改内容
在这里插入图片描述
再次运行代码
在这里插入图片描述

3. Class与加载方式

3.1 所有类型的Class对象

要想了解一个类,必须先要获取到该类的字节码文件对象.

在Java中,每一个字节码文件,被加载到内存后,都存在一个对应的Class类型的对象
在这里插入图片描述

3.2 得到Class的几种方式(将类加载到内存的方式)

  1. 如果在编写代码时, 指导类的名称, 且类已经存在, 可以通过
  • 包名.类名.class 得到一个类的 类对象
  1. 如果拥有类的对象, 可以通过
  • Class 对象.getClass() 得到一个类的 类对象
  1. 如果在编写代码时, 知道类的名称 , 可以通过
  • Class.forName(包名+类名): 得到一个类的 类对象

上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存 !如果类已经在内存中存在, 不 会重复加载, 而是重复利用 !

举例:

/**
 * @Author: 爱摸鱼的TT~
 * @Description:  Class与类加载方式(Class:类类型)
 * 得到Class的几种方式
 * @Date Created in 2021-03-22 14:39
 * @Modified By:
 */
public class Demo_2 {
    public static void main(String[] args) throws ClassNotFoundException {
        // 第一种方式: 通过包名.类名.class 加载类
        Class<Person> p = fanshe.com.java.Person.class;
        System.out.println(p);

        // 第二种方式:通过类的对象获取类的信息
        Person p2 = new Person();
        Class a = p2.getClass();
        System.out.println(a);

        // 第三种方式:通过 Class.forName(包名+类名) 加载类【动态加载】
        Class p3 = Class.forName("fanshe.com.java.Person");
        System.out.println(p3);

        // 上述的三种方式, 在调用时, 如果类在内存中不存在, 则会加载到内存 ! 如果类已经在内存中存在, 不 会重复加载, 而是重复利用 !
    }
}

运行结果:
在这里插入图片描述

当把Person类删除后
在这里插入图片描述

所以一些框架可以通过这种方法将类加载到内存中去:
在这里插入图片描述

4. 获取Constructor(构造方法)

4.1 通过class对象-获取一个类的构造方法

  1. 通过指定的参数类型, 获取指定的单个构造方法
    例如:

构造方法如下:

Person(String name,int age) 

得到这个构造方法的代码如下:

Constructor c = p.getClass().getConstructor(String.class,int.class);
  1. 获取构造方法数组
getConstructors(); 
  1. 获取所有权限的单个构造方法
getDeclaredConstructor(参数类型的class对象数组) 
  1. 获取所有权限的构造方法数组
getDeclaredConstructors();

4.2 Constructor创建对象

newInstance(Object... para) 

调用这个构造方法, 把对应的对象创建出来

参数: 是一个Object类型可变参数, 传递的参数顺序 必须匹配构造方法中形式参数列表的顺序!

setAccessible(boolean flag) 

如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

4.3 反射技术打破封装的例子

举例:

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

/**
 * @Author: 爱摸鱼的TT~
 * @Description: 通过class对象 获取一个类的构造方法
 * @Date Created in 2021-03-22 15:23
 * @Modified By:
 */
public class Demo_3 {
    public static void main(String[] args) throws Exception {
        // 1. 将类加载到内存【动态加载】
        Class<Person> a = (Class<Person>) Class.forName("fanshe.com.java.Person");

        // 2. 使用Constructor 获取一个类的构造方法
        // (1) 找到无参构造方法
        Constructor<Person> c1 = a.getConstructor();
        // 3. 使用无参构造方法,创建对象
        Person p1 = c1.newInstance();
        System.out.println(p1); // Person{name='null', age=0}

        // (2) 找到含有参数的构造方法
        Constructor<Person> c2 = a.getConstructor(String.class, int.class);
        //Constructor<Person> c3 = a.getConstructor(new Class[]{String.class,int.class}); // 通过数组传入可变参数
        // 3. 使用有参构造方法,创建对象
        Person p2 = c2.newInstance("爱摸鱼的TT~", 18);
        System.out.println(p2); // Person{name='爱摸鱼的TT~', age=18}

        // (3) 有访问权限的(private)的构造方法
        Constructor<Person> c3 = a.getDeclaredConstructor(String.class);
        // 可以访问任何权限的方法(可以通过外部调用私有的方法)
        c3.setAccessible(true);
        // 3. 使用私有的构造方法,创建对象
        Person p3 = c3.newInstance("爱摸鱼的TT");
        System.out.println(p3); // Person{name='爱摸鱼的TT', age=19}
    }
}

5. 获取Method

通过反射找到类中的方法,再通过对象执行类的方法。

5.1 通过class对象-获取一个类的方法

  1. getMethod(String methodName , class… clss)

根据参数列表的类型和方法名, 得到一个方法(public修饰的)

  1. getMethods();

得到一个类的所有方法 (public修饰的)

  1. getDeclaredMethod(String methodName , class… clss)

根据参数列表的类型和方法名, 得到一个方法(除继承以外所有的:包含私有, 共有, 保护, 默认)

  1. getDeclaredMethods();

得到一个类的所有方法 (除继承以外所有的:包含私有, 共有, 保护, 默认)

5.2 Method执行方法

invoke(Object o,Object… para) :

  • 参数1. 要调用方法的对象
  • 参数2. 要传递的参数列表

getName()

  • 获取方法的方法名称

setAccessible(boolean flag)

  • 如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的方法)

5.3 举例

import javax.management.DynamicMBean;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
 * @Author: 爱摸鱼的TT~
 * @Description:通过class对象 获取一个类的普通方法
 * @Date Created in 2021-03-22 16:05
 * @Modified By:
 */
public class Demo_4 {
    public static void main(String[] args) throws Exception {
        // 1. 将类加载到内存
        Class aClass = Class.forName("fanshe.com.java.Person_1");
        // 2. 获取构造方法
        Constructor<?> c1 = aClass.getConstructor();
        // 3. 创建对象
        Object o = c1.newInstance();
        // 4.获取类的普通方法
        Method m1 = aClass.getMethod("setName", String.class); // 这是获取公有的方法
        Method m2 = aClass.getDeclaredMethod("setAge", int.class); // 这里获取私有的方法
        m2.setAccessible(true);
        //参数1:哪个对象要执行set**方法
        //参数2:调用方法时传递的参数:0-n
        m1.invoke(o,"爱摸鱼的TT");
        m2.invoke(o,18);
        System.out.println(o);
    }
}

运行结果:
在这里插入图片描述

6. 获取Filed(属性)

6.1 通过class对象-获取一个类的属性

  1. getDeclaredField(String filedName)
  • 根据属性的名称, 获取一个属性对象 (所有属性)
  1. getDeclaredFields()
  • 获取所有属性
  1. getField(String filedName)
  • 根据属性的名称, 获取一个属性对象 (public属性)
  1. getFields()
  • 获取所有属性 (public)

6.2 Filed的属性的对象类型

常用方法:

  1. get(Object o ); 获取指定对象的此属性值
  • 参数: 要获取属性的对象
  1. set(Object o , Object value);设置指定对象的属性的值
  • 参数1. 要设置属性值的 对象
  • 参数2. 要设置的值
  1. getName() 获取属性的名称

  2. setAccessible(boolean flag)

  • 如果flag为true 则表示忽略访问权限检查 !(可以访问任何权限的属性)
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

/**
 * @Author: 爱摸鱼的TT~
 * @Description:通过class对象 获取一个类的属性
 * @Date Created in 2021-03-22 16:05
 * @Modified By:
 */
public class Demo_5 {
    public static void main(String[] args) throws Exception {
        // 1. 将类加载到内存
        Class aClass = Class.forName("fanshe.com.java.Person_2");
        // 2. 获取构造方法
        Constructor c1 = aClass.getConstructor();
        // 3. 创建对象
        Object o = c1.newInstance();
        // 4.获取类的属性
        Field name_1 = aClass.getField("phoneNumber");
        // 5. 设置属性值
        name_1.set(o,"13686644128");

        // 获取私有属性的方式
        Field name_2 = aClass.getDeclaredField("name");
        name_2.setAccessible(true);
        name_2.set(o,"爱摸鱼的TT~");
        System.out.println(o);
    }
}

运行结果:
在这里插入图片描述

7. 获取注解信息

通过反射获取注解的内容

7.1 获取类/属性/方法的全部注解对象

Annotation[] annotations01 = Class/Field/Method.getAnnotations(); 
    for (Annotation annotation : annotations01) { 
    System.out.println(annotation); 
}

7.2 根据类型获取类/属性/方法的注解对象

注解类型 对象名 = (注解类型) c.getAnnotation(注解类型.class);
在这里插入图片描述

7.3 举例

在这里插入图片描述
TableAnnotation.java

import java.lang.annotation.*;
 
@Target(ElementType.TYPE)               // 允许使用在类上
@Retention(RetentionPolicy.RUNTIME)     // 持久化策略
@Documented                             // 允许写在文档中
public @interface TableAnnotation {
    /**
     * 用于标注类对应的表格名称(与数据库关联)
     * @return
     */
    String value();
}

ColumnAnnotation.java

import java.lang.annotation.*;
 
@Target(ElementType.FIELD)// 允许使用在属性上
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface ColumnAnnotation {
    /**
     * 描述列名
     * @return
     */
    String columnName();
 
    /**
     * 描述类型
     * @return
     */
    String type();
 
    /**
     * 描述数据的长度
     * @return
     */
    String length();
}

Book.java

import java.util.Objects;
// 内部原理:通过反射找到类,并找到类上的注解,最终基于注解中的内容与数据库中的表格进行绑定
// 这里虽然加上了自定义的四个注解,但是缺少背后与数据库关联的逻辑,所以没有实际作用
@TableAnnotation("test_Book")// 通过注解描述 这个类和数据库中的Book对应
public class Book {
    @ColumnAnnotation(columnName = "id", type = "int", length = "11")
    private int id;
    @ColumnAnnotation(columnName = "name", type = "varchar", length = "50")
    private String name;
    @ColumnAnnotation(columnName = "info", type = "varchar", length = "1000")
    private String info;
 
    public Book(int id, String name, String info) {
        this.id = id;
        this.name = name;
        this.info = info;
    }
 
    public Book() {
    }
 
    public int getId() {
        return id;
    }
 
    public void setId(int id) {
        this.id = id;
    }
 
    public String getName() {
        return name;
    }
 
    public void setName(String name) {
        this.name = name;
    }
 
    public String getInfo() {
        return info;
    }
 
    public void setInfo(String info) {
        this.info = info;
    }
 
    @Override
    public String toString() {
        return "Book{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", info='" + info + '\'' +
                '}';
    }
 
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Book book = (Book) o;
        return id == book.id &&
                Objects.equals(name, book.name) &&
                Objects.equals(info, book.info);
    }
 
    @Override
    public int hashCode() {
        return Objects.hash(id, name, info);
    }
}

Demo.java

import java.io.File;
import java.lang.reflect.Field;
import java.text.Annotation;
 
public class Demo {
    public static void main(String[] args) throws ClassNotFoundException {
        Class c = Class.forName("com.kaikeba.demo.Book");
        TableAnnotation ta = (TableAnnotation) c.getAnnotation(TableAnnotation.class);
        String value = ta.value();
        System.out.println("表名:" + value);      // 类名作为数据库中表格的名称
 
        Field[] fs = c.getDeclaredFields();       // 属性名作为数据库中的字段名称
        for(Field f : fs) {
            ColumnAnnotation ca = f.getAnnotation(ColumnAnnotation.class);
            System.out.println(f.getName() + "属性,对应数据库中的字段:" + ca.columnName() + ";数据类型:" + ca.type() + ";数据长度:"  + ca.length());
        }
 
    }
}

运行结果:
在这里插入图片描述

一些ORM框架的原理:当使用提供的注解时,框架通过找到类,来获取注解的值(类名以及属性),然后就可以根据值在数据库中创建表(将类名与属性作为表格与字段的名称)

四、内省

1. 简介

基于反射 , java所提供的一套应用到JavaBean的API

  • 一个定义在包中的类 ,
  • 拥有无参构造器
  • 所有属性私有,
  • 所有属性提供get/set方法
  • 实现了序列化接口

这种类, 我们称其为 bean类 . (一些没有业务逻辑的类

Java提供了一套java.beans包的API, 对于反射的操作, 进行了封装

2. Introspector

获取Bean类信息

方法:

  • BeanInfo getBeanInfo(Class cls)
  • 通过传入的类信息, 得到这个Bean类的封装对象 .

3. BeanInfo

常用的方法:

  • MethodDescriptor[] getPropertyDescriptors():
  • 获取bean类的 get/set方法 数组

4. MethodDescriptor

常用方法:

1.Method getReadMethod();

  • 获取一个get方法

2.Method getWriteMethod();

  • 获取一个set方法

有可能返回null 注意 ,加判断 !

5. 内省的机制

在这里插入图片描述
代码:
Express.java

import java.io.Serializable;

/**
 * @Author: 爱摸鱼的TT~
 * @Description: 设计Bean类(规则要遵守)
 * 需要注意Boolean类型 获得get/set方法结果是get-is,set-set。
 * @Date Created in 2021-03-23 9:02
 * @Modified By:
 */
public class Express implements Serializable {
    private String name;
    private String number;
    private String phoneNumber;
    private String address;
    private boolean flag; // 需要注意boolean类型获得到的方法

    public Express() {
    }

    public boolean isFlag() {
        return flag;
    }

    public void setFlag(boolean flag) {
        this.flag = flag;
    }

    public Express(String name, String number, String phoneNumber, String address) {
        this.name = name;
        this.number = number;
        this.phoneNumber = phoneNumber;
        this.address = address;
    }

    public String getName() {
        return name;
    }

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

    public String getNumber() {
        return number;
    }

    public void setNumber(String number) {
        this.number = number;
    }

    public String getPhoneNumber() {
        return phoneNumber;
    }

    public void setPhoneNumber(String phoneNumber) {
        this.phoneNumber = phoneNumber;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    @Override
    public String toString() {
        return "Express{" +
                "name='" + name + '\'' +
                ", number='" + number + '\'' +
                ", phoneNumber='" + phoneNumber + '\'' +
                ", address='" + address + '\'' +
                '}';
    }
}

Demo_1.java

import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.function.DoubleToIntFunction;

/**
 * @Author: 爱摸鱼的TT~
 * @Description:
 * @Date Created in 2021-03-23 9:05
 * @Modified By:
 */
public class Demo_1 {
    public static void main(String[] args) throws IntrospectionException {
        // 获得类信息
        Class c = Express.class;
        // 获取Bean类信息
        BeanInfo bi = Introspector.getBeanInfo(c);
        // 获取Bean类的get/set数组信息
        PropertyDescriptor[] pds = bi.getPropertyDescriptors();
        for(PropertyDescriptor pd: pds){
            // 获取一个get/set方法信息
            Method get = pd.getReadMethod();
            Method set = pd.getWriteMethod();
            //获得get/set方法
            System.out.println(get);
            System.out.println(set);
            // 获得属性的名称
            System.out.println(pd.getName());
            // 获得属性的类型
            System.out.println(pd.getPropertyType());
        }
    }
}

运行结果:
在这里插入图片描述

通过内省机制,可以更快速的操作Bean对象获得Getter和Setter方法(相比于用基础的反射操作)

相信读者还是会对什么是JavaBean有些陌生,我这里转载一篇 什么是javabean及其用法

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

涛涛同学debug

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值