Java学习笔记 反射

反射

  • JAVA反射机制是在运行状态中,获取任意一个类的结构 , 创建对象 , 得到方法,执行方法 , 属性。
1,类加载器
  • Java类加载器(Java Classloader)是Java运行时环境(Java Runtime Environment)的一部分, 负责动态加载Java类到Java虚拟机的内存空间中。

  • java默认有三种类加载器,BootstrapClassLoader、ExtensionClassLoader、App ClassLoader。

    1. BootstrapClassLoader(引导启动类加载器): 嵌在JVM内核中的加载器,该加载器是用C++语言写 的,主要负载加载JAVA_HOME/lib下的类库,引 导启动类加载器无法被应用程序直接使用。

    2. ExtensionClassLoader(扩展类加载器): ExtensionClassLoader是用JAVA编写,且它的父类加载 器是Bootstrap。 是由sun.misc.Launcher$ExtClassLoader实现的,主要加载 JAVA_HOME/lib/ext目录中的类 库。 它的父加载器是BootstrapClassLoader

    3. App ClassLoader(应用类加载器): App ClassLoader是应用程序类加载器,负责加载应用程序classpath目录下的所有jar和class文 件。 它的父加载器为ExtensionClassLoader

  • 类通常是按需加载,即第一次使用该类时才加载。由于有了类加载器,Java运行时系统不需要知道文件与 文件系统。学习类加载器时,掌握Java的委派概念很重要。

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

2, 获取Class对象
  • 要想了解一个类,必须先要获取到该类的字节码文件对象. 在Java中,每一个字节码文件,被夹在到内存 后,都存在一个对应的Class类型的对象

先给出下面所有代码都在操作的类

public class Person {
    private String name;
    private int age;
    public String phoneNumber;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    private Person(String name) {
        this.name = name;
        age = 88;
    }

    public Person() {
    }

    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", phoneNumber='" + phoneNumber + '\'' +
                '}';
    }

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

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    private void setNameAndAge(String name, int age){
        this.name = name;
        this.age  = age;
    }
}

public class Demo2 {
    /**
     * 获得Class的三种方式
     *
     * @param args
     */
    public static void main(String[] args) throws ClassNotFoundException {
        //第一种方式,通过包名.类名.class 加载类
        Class c = Person.class;
        System.out.println(c);

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

        //第三种方式,若不知道类的名称,可以通过:
        Class c3 = Class.forName("day37.Person");
        System.out.println(c3);
    }
}
  • 上述三种方式,在调用时,如果类在内存中不存在,则会将类加载至内存中,如果类已经在内存中存在,则不会重复加载,而是重复利用
3,获取Constructor
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Demo3 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        //加载类
        Class pClass = Class.forName("day37.Person");

        //1,获取公有的无参构造方法(默认public)
        Constructor constructor1 = pClass.getConstructor();
        Person p = (Person) constructor1.newInstance();
        System.out.println(p);

        //2,获取公有的有参构造方法
        Constructor constructor2 = pClass.getConstructor(String.class , int.class);
        Person p2 = (Person) constructor2.newInstance("张三", 18);
        System.out.println(p2);

        //3,获取private的构造方法
        Constructor constructor3 = pClass.getDeclaredConstructor(String.class);
        constructor3.setAccessible(true);
        Person p3 = (Person) constructor3.newInstance("李四");
        System.out.println(p3);
    }
}
4,获取method
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class Demo4 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        //加载类
        Class c = Class.forName("day37.Person");

        //1,默认获得public修饰的方法

        //调用构造方法初始化
        Constructor constructor = c.getConstructor();

        //获得对象
        Object o = constructor.newInstance();

        //获得public方法
        Method setName = c.getMethod("setName", String.class);

        //调用方法,参数1:该对象;参数2-n:调用该方法时要传的参数
        setName.invoke(o,"张三");
        System.out.println(o);

        //2,获得所有权限的方法
        Constructor constructor1 = c.getConstructor();
        Object o2 = constructor1.newInstance();

        Method setNameAndAge = c.getDeclaredMethod("setNameAndAge", String.class, int.class);
        //消除该方法的访问权限
        setNameAndAge.setAccessible(true);
        setNameAndAge.invoke(o2, "李四", 18);

        System.out.println(o2);
    }
}
5,获取属性Field
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;

public class Demo5 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException, NoSuchFieldException {
        //三部曲,用无参构造方法创建对象
        Class c = Class.forName("day37.Person");
        Constructor ct = c.getConstructor();
        Object o = ct.newInstance();

        //获取public的Field
        //给phoneNumber设置值
        Field phoneNumber = c.getField("phoneNumber");
        phoneNumber.set(o, "13843838438");
        System.out.println(o);

        //获取各种权限的Field
        Field name = c.getDeclaredField("name");
        name.setAccessible(true);
        name.set(o, "李四");
        System.out.println(o);
    }
}
6,获取注解

先自定义两个注解ColumnAnnotation与TableAnnotation和一个测试类Book

package day37.demoAnnotation;

import java.lang.annotation.*;

/**
 * 1,可以加载到文档里
 * 2,使用范围:属性
 * 3,持久化策略:RUNTIME
 */
@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ColumnAnnotation {
    /**
     * 描述一个表中的列名
     * @return
     */
    String columnName();

    /**
     * 描述一列的类型
     * @return
     */
    String type();

    /**
     * 描述一列长度
     * @return
     */
    String length();
}

package day37.demoAnnotation;

import java.lang.annotation.*;

/**
 * 什么意思呢?
 * 这个注解的用法
 * 1,允许继承
 * 2,可以加载到文件里
 * 3,注解允许使用的类型是:类
 * 4,持久化策略:RUNTIME
 */
@Inherited
@Documented
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface TableAnnotation {
    /**
     * 用于标注类对应的表格名称
     * @return
     */
    String value();
}

package day37.demoAnnotation;

import java.util.Objects;

@TableAnnotation("test_book")
public class Book {
    @ColumnAnnotation(columnName = "bid", type = "int",length = "11")
    private int id;
    @ColumnAnnotation(columnName = "bname", type = "varchar",length = "50")
    private String name;
    @ColumnAnnotation(columnName = "binfo", type = "varchar",length = "1000")
    private String 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);
    }

    @Override
    public String
    toString() {
        return "Book{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", info='" + info + '\'' +
                '}';
    }

    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;
    }

    public Book() {
    }

    public Book(int id, String name, String info) {
        this.id = id;
        this.name = name;
        this.info = info;
    }
}

测试类:

import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;

public class Demo {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class c = Class.forName("day37.demoAnnotation.Book");

        TableAnnotation ta = (TableAnnotation) c.getAnnotation(TableAnnotation.class);
        String value = ta.value();
        System.out.println("表名:" + value);

        Field[] dfs = c.getDeclaredFields();
        for (Field df : dfs) {
            ColumnAnnotation ca = df.getAnnotation(ColumnAnnotation.class);
            System.out.println(df.getName()+" 属性,对应数据库中字段;列名:"+ca.columnName()+",数据类型:"+ca.type()+",长度:"+ca.length());
        }

    }
}
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值