Java Reflection

本文深入探讨了Java的反射机制,对比了静态语言与动态语言的区别,阐述了反射在运行时如何获取类信息、创建对象、操作方法和属性。还讨论了类加载器的作用、类的初始化时机以及泛型在反射中的处理。同时,通过实例展示了如何通过反射调用方法、修改私有属性,并分析了反射对性能的影响。最后,简要总结了ORM(对象关系映射)的概念。
摘要由CSDN通过智能技术生成

什么是Java的反射机制

静态语言与动态语言

  动态语言是在运行时可以改变自身结构的语言。

/*JavaScript*/
/*可以在运行的时候改变某些自身结构*/
/*程序在运行时可以改变x的值*/
function f() {
    var x = "var a= 3,var b = 5;alert(a+b)"
    eval(x);
}

  与动态语言相对的是静态语言,即在运行时结构不可变。
  Java不是动态语言但可被称为时准动态语言,可以利用反射机制获得类似动态语言的特性,与此同时增加了不安全性。
  反射机制允许程序在执行过程中借助反射的API取得任何类的内部信息,包括这个类的类名、接口、方法、字段、属性等等,并可以直接操纵类的属性以及信息。

正常方式:先引入需要的“包类”名称===》通过new实例化===》取得实例化对象
反射方式:先实例化对象===》然后getClass()方法===》得到完整的"包类"名称

反射机制的研究以及应用

  在运行时判断任意一个对象所属的类;构造任意一个类的对象;判断任意成员的变量和方法;在运行时获取泛型信息;在运行时调用任意一个对象的变量和方法;在运行时处理注解;生成动态代理。

反射机制的优缺点

  优点:可以实现动态创建对象和编译,具有很大的灵活性
  缺点:对性能有影响,反射机制时一种解释操作,我们需要告诉虚拟机(JVM),我们总希望做什么并且它满足我们的要求。所以这种机制要慢于直接致性的操作。

什么是反射,对实体类的简单定义

package com.ly.reflection;
@SuppressWarnings("all")
public class test01 extends Object{
    public static void main(String[] args) throws ClassNotFoundException {
        /*通过反射获取类的class对象*/
        Class aClass = Class.forName("com.ly.reflection.User");
        System.out.println(aClass);

        Class aClass1 = Class.forName("com.ly.reflection.User");
        Class aClass2 = Class.forName("com.ly.reflection.User");
        Class aClass3 = Class.forName("com.ly.reflection.User");
        /*一个类在内存中只有一个class对象*/
        /*一个类被加载后,类的整个结构都会被封装在class结构中*/
        System.out.println(aClass1.hashCode());
        System.out.println(aClass2.hashCode());
        System.out.println(aClass3.hashCode());
    }
}

/*构造一个实体类:pojo对象 entity实体*/
class User {
    private String name;
    private int id;
    private int age;

    public User() {
    }

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

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

    public String getName() {
        return name;
    }

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

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public int getAge() {
        return age;
    }

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

Class类

public final Class getClass()

  以上方法的返回值类型是一个Class类,此类为Java反射的源头,从实际情况的运行结果来看,可以通过对象反射求出类的名称。
  Class本身也是一个类;只能有系统创建;一个加载的类在JVm中只有一个实例;一个CLass对应一个加载到JVM的.class文件;每个类的实例都会记得自己是由哪个Class实例所生成;通过Class可以完整得到一个类中所有被加载的结构;Class是Reflection的根源,针对任何你想动态加载、运行的类,唯有先获取相应的Class对象。

获取Class类的实例

  若已知具体的类,可以通过类的class属性获取,这种方法是安全可靠,程序性能最高。

Class aClass = Person.class;

  已知莫个类的实例,调用该实例的getClass()方法以获取Class对象

Class aClass1 = person.getClass();

  已知一个类的全类名,在该类名路径下,可以通过Class类的静态方法forName()获取,可能抛出ClassNotFoundException

Class aClass2 = Class.forName("demo01.Student");

  针对内置数据类型可以直接用类名.Type
  还可以利用ClassLoader

测试Class类的创建方法有哪些

package com.ly.reflection;

/*测试class类的创建方式有哪些*/
/*@SuppressWarnings("all")*/
public class Test03 {
    public static void main(String[] args) throws ClassNotFoundException {
        Person person = new Student();
        System.out.println("这个人是:" + person.name);
        /*方式一:通过对象获得*/
        Class aClass = person.getClass();
        System.out.println(aClass.hashCode());
        /*方式二:forname:获得*/
        Class aClass1 = Class.forName("com.ly.reflection.Student");
        System.out.println(aClass1.hashCode());
        /*方式三:这是一个静态属性,通过类名.class获得*/
        Class aClass2 = Student.class;
        System.out.println(aClass2.hashCode());
        /*方式四:基本内置的包装类都有一个Type属性*/
        Class aClass3 = Integer.TYPE;
        System.out.println(aClass3);
        /*获得父类的类型*/
        Class aClass4 = aClass.getSuperclass();
        System.out.println(aClass4);
    }
}

class Person {
    public String name;

    public Person() {
    }

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


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

class Student extends Person {
    public Student() {
        this.name = "学生";
    }
}

class Teacher extends Person {
    public Teacher() {
        this.name = "老师";
    }
}

那些类型可以有Class对象

  class:外部类,成员,局部内部类,匿名内部类;interface:接口;[]:数组;enum:枚举;annotation:注解@interface;primitive type:基本数据类型;void

package com.ly.reflection;

import java.lang.annotation.ElementType;

/*所有类型的class*/
public class Test04 {
    public static void main(String[] args) {
        Class c1 = Object.class;/*类*/
        Class c2 = Comparable.class;/*接口*/
        Class c3 = String[].class;/*一维数组*/
        Class c4 = int[][].class;/*二维数组*/
        Class c5 = Override.class;/*注解*/
        Class c6 = ElementType.class;/*枚举*/
        Class c7 = Integer.class;/*基本数组类型*/
        Class c8 = void.class;/*void,无类型*/
        Class c9 = Class.class;/*Class*/

        System.out.println("----------------开始-----------------");
        System.out.println(c1);
        System.out.println(c2);
        System.out.println(c3);
        System.out.println(c4);
        System.out.println(c5);
        System.out.println(c6);
        System.out.println(c7);
        System.out.println(c8);
        System.out.println(c9);

        /*只要元素类型与维度一样,就是同一个Class*/
        int[] a = new int[10];
        int[] b = new int[100];
        int[] c = new int[1000];
        System.out.println(a.getClass().hashCode());
        System.out.println(b.getClass().hashCode());
        System.out.println(c.getClass().hashCode());
        System.out.println("----------------结束-----------------");
    }
}

----------------开始-----------------
class java.lang.Object
interface java.lang.Comparable
class [Ljava.lang.String;
class [[I
interface java.lang.Override
class java.lang.annotation.ElementType
class java.lang.Integer
void
class java.lang.Class
460141958
460141958
460141958
----------------结束-----------------

Java内存分析

类的加载与ClassLoader

  加载:将class文件字节码加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后生成一个代表这个类的java,lang.Class对象
  链接:将Java类的二进制代码合并到JVM的运行情况之中的过程。
  初始化:执行类构造器< clinit>()的过程。类构造器是构造类信息的,不是构造该类对象的构造器;当初始化一个类的时候,如果发现其父类还没有进行初始化,则需要先触发其父类的初始化;JVM会保证一个类的< clinit>()方法在多线程环境中被正确的加锁和同步。

package com.ly.reflection;

/*@SuppressWarnings("all")*/
public class Test05 {
    public static void main(String[] args) {
        System.out.println("-------------------------开始-------------------------");
        A a = new A();
        System.out.println(A.m);
        System.out.println("-------------------------结束-------------------------");
        /**
         * 1.加载到内存,会产生一个类对应Class对象
         * 2.链接,链接结束后m=0
         * 3.初始化
         *       <clinit>(){
         *           SYstem.out.println("A类静态代码快初始化")
         *           m = 300;
         *           m = 100;
         *       }
         *       m = 100
         */
    }
}
class A {
    static int m = 100;
    static {
        System.out.println("A类静态代码块初始化");
        m = 300;
    }
    public A() {
        System.out.println("A类的无参构造初始化");
    }
}


-------------------------开始-------------------------
A类静态代码块初始化
A类的无参构造初始化
300
-------------------------结束-------------------------

什么时候会发生类的初始化

  一定会发生类的初始化(类的主动引用):当虚拟机启动,先初始化main方法所在的类;new一个类的对象;调用静态的静态成员(除了final常量)和静态方法;使用java.lang.relext包的方法对类反复调用;当初始化一个类,如果其父类没有被初始化,则会先初始化他的父类。
  当访问一个静态时,只有真正声这个域的类才会被初始化。如:当通过子类引用父类的静态变量,不会导致子类初始化;通过数组定义类引用,不会触发此类的初始化;引用常量不会触发此类的初始化(常量在链接阶段就已经存入调用类的常量池中了)

package com.ly.reflection;
/*测试类什么时候会初始化*/
public class Test06 {
    static {
        System.out.println("Main被加载");
    }
    public static void main(String[] args) throws ClassNotFoundException {
        System.out.println("-------------------------开始-------------------------");
        /*1.主动引用*//*
        Son son = new Son();
        *//*反射也会产生主动引用*/
        /*Class.forName("cpm.ly.relection.Son");*/
        /*不会产生类的引用的方法*/
        /*System.out.println(Son.b);*/
        /*Son [] array = new Son[5];*/
        System.out.println( );
        System.out.println("-------------------------结束-------------------------");
    }
}
class Father {
    static int b = 2;

    static {
        System.out.println("父类被加载");
    }
}
class Son extends Father {
    static final int M = 1;
    static int m = 100;
    static {
        System.out.println("子类被加载");
        m = 300;
    }
}

类加载器的作用

  类加载的作用:将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为放发区中类数据的访问入口。
  类缓存:标准的JavaSE类加载器可以按照要求查找类,但一旦某个类被加载到类加载器中,它将维持加载(缓存)一段时间。不过JVM垃圾回收机制可以回收这些Class对象。

源文件(.java文件)—>Java编译器—>字节码(.class文件)—>类加载器—>字节码校验器—>解释器—>操作系统平台 **

  JVM规范定义一下类型的类加载器:引导类加载器(Bootstap Classloader),时JVM自带的,负责Java平台核心库,无法直接获取;扩展类加载器(Extension Classloader),负责jre/lib/ext目录的包加载到jar包装入工作库;系统类加载器(System Classloader/Application Classloader),将项目里所指定的一些架包加进去

package com.ly.reflection;

/*@SuppressWarnings("all")*/
public class Test07 {
    public static void main(String[] args) throws ClassNotFoundException {
        System.out.println("-------------------------开始-------------------------");
        //获取系统类的加载器
        ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
        System.out.println(systemClassLoader);
        //获取系统类加载器的父类加载器--->扩展类加载器
        ClassLoader parent = systemClassLoader.getParent();
        System.out.println(parent);
        /*获取扩展类加载器的父类加载器--->根加载器(c/c++)*/
        ClassLoader parent1 = parent.getParent();
        System.out.println(parent1);
        /*测试当前类是哪个加载器加载的*/
        ClassLoader classLoader = Class.forName("com.ly.reflection.Test07").getClassLoader();
        System.out.println(classLoader);
        /*测试jdk内部的类是谁加载的*/
        classLoader = Class.forName("java.lang.Object").getClassLoader();
        System.out.println(classLoader);
        /*如果获得系统类加载器可以加载的路径*/
        System.out.println(System.getProperty("java.class.path"));
        /*双亲委派机制:java.lang.String*/
       /* C:\Program Files\Java\jdk1.8.0_191\jre\lib\charsets.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\deploy.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\access-bridge-64.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\cldrdata.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\dnsns.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\jaccess.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\jfxrt.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\localedata.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\nashorn.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunec.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunjce_provider.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunmscapi.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunpkcs11.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\zipfs.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\javaws.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\jce.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\jfr.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\jfxswt.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\jsse.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\management-agent.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\plugin.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\resources.jar;
       C:\Program Files\Java\jdk1.8.0_191\jre\lib\rt.jar;
       D:\Project\Annotation\out\production\Annotation;
       D:\IntelliJ IDEA 2020.3\lib\idea_rt.jar*/
        System.out.println("-------------------------结束-------------------------");
    }
}

-------------------------开始-------------------------
sun.misc.Launcher$AppClassLoader@18b4aac2
sun.misc.Launcher$ExtClassLoader@1b6d3586
null
sun.misc.Launcher$AppClassLoader@18b4aac2
null
C:\Program Files\Java\jdk1.8.0_191\jre\lib\charsets.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\deploy.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\access-bridge-64.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\cldrdata.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\dnsns.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\jaccess.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\jfxrt.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\localedata.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\nashorn.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunec.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunjce_provider.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunmscapi.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\sunpkcs11.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\ext\zipfs.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\javaws.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\jce.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\jfr.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\jfxswt.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\jsse.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\management-agent.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\plugin.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\resources.jar;C:\Program Files\Java\jdk1.8.0_191\jre\lib\rt.jar;D:\Project\Annotation\out\production\Annotation;D:\IntelliJ IDEA 2020.3\lib\idea_rt.jar
-------------------------结束-------------------------

创建运行时类的对象

获取运行时类的完整结构

  通过案涉获取运行时类的完整结构:Field、Method、Constructor、Superclass、Interface、Annotation

package com.ly.reflection;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;

/*@SuppressWarnings("all")*/
/*获取类的信息*/
public class Test08 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {
        System.out.println("-------------------------开始-------------------------");
        /*简单字符串可以通过web传递*/
        Class aClass = Class.forName("com.ly.reflection.User");
        /*获得类的名字*/
        System.out.println(aClass.getName());/*获取包名,类名*/
        System.out.println(aClass.getSimpleName());/*获得类名*/
        /*获得类的属性*/
        Field[] fields = aClass.getFields();/*只能找到public方法*/
        fields = aClass.getDeclaredFields();/*找到全部的属性*/
        for (Field field : fields) {
            System.out.println(field);
        }
        /*获得属性指定的值*/
        Field name = aClass.getDeclaredField("name");
        System.out.println(name);
        /*获得类的方法*/
        Method[] methods = aClass.getMethods();/*获得本类及其父类的所有public方法*/
        for (Method method : methods) {
            System.out.println("正常的" + method);
        }
        methods = aClass.getDeclaredMethods();/*获得本类的所有方法*/
        for (Method method : methods) {
            System.out.println("getDeclaredMethods" + method);
        }
        /*获得指定方法*/
        /*重载*/
        Method getName = aClass.getMethod("getName", null);
        Method setName = aClass.getMethod("setName", String.class);
        System.out.println(getName);
        System.out.println(setName);
        /*获得指定的构造器*/
        Constructor[] constructors = aClass.getConstructors();
        for (Constructor constructor : constructors) {
            System.out.println(constructor);
        }
        constructors = aClass.getDeclaredConstructors();
        for (Constructor constructor : constructors) {
            System.out.println("#" + constructor);
        }
        /*获得指定的构造器*/
        Constructor declaredConstructor = aClass.getDeclaredConstructor(String.class, int.class, int.class);
        System.out.println("指定:" + declaredConstructor);
        System.out.println("-------------------------结束-------------------------");
    }
}

-------------------------开始-------------------------
com.ly.reflection.User
User
private java.lang.String com.ly.reflection.User.name
private int com.ly.reflection.User.id
private int com.ly.reflection.User.age
private java.lang.String com.ly.reflection.User.name
正常的public java.lang.String com.ly.reflection.User.toString()
正常的public java.lang.String com.ly.reflection.User.getName()
正常的public int com.ly.reflection.User.getId()
正常的public void com.ly.reflection.User.setName(java.lang.String)
正常的public void com.ly.reflection.User.setId(int)
正常的public int com.ly.reflection.User.getAge()
正常的public void com.ly.reflection.User.setAge(int)
正常的public final void java.lang.Object.wait() throws java.lang.InterruptedException
正常的public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
正常的public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
正常的public boolean java.lang.Object.equals(java.lang.Object)
正常的public native int java.lang.Object.hashCode()
正常的public final native java.lang.Class java.lang.Object.getClass()
正常的public final native void java.lang.Object.notify()
正常的public final native void java.lang.Object.notifyAll()
getDeclaredMethodspublic java.lang.String com.ly.reflection.User.toString()
getDeclaredMethodspublic java.lang.String com.ly.reflection.User.getName()
getDeclaredMethodspublic int com.ly.reflection.User.getId()
getDeclaredMethodspublic void com.ly.reflection.User.setName(java.lang.String)
getDeclaredMethodspublic void com.ly.reflection.User.setId(int)
getDeclaredMethodspublic int com.ly.reflection.User.getAge()
getDeclaredMethodspublic void com.ly.reflection.User.setAge(int)
public java.lang.String com.ly.reflection.User.getName()
public void com.ly.reflection.User.setName(java.lang.String)
public com.ly.reflection.User()
public com.ly.reflection.User(java.lang.String,int,int)
#public com.ly.reflection.User()
#public com.ly.reflection.User(java.lang.String,int,int)
指定:public com.ly.reflection.User(java.lang.String,int,int)
-------------------------结束-------------------------

进程已结束,退出代码0

简单的小结

  在实际的操作中,获得类的信息的操作代码,并不会经常开发。
  熟悉java,lang.reflect包的作用,反射机制很重要。
  如何取得属性、方法、构造器的名称,修饰符等等。

*如何获得通过反射来创建一个对象 *

特殊标记:at com.ly.reflection.Test09.main(Test09.java:22)

package com.ly.reflection;

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

/*@SuppressWarnings("all")*/
public class Test09 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        System.out.println("-------------------------开始-------------------------");
        /*获得Class对象*/
        Class aClass = Class.forName("com.ly.reflection.User");
        /*构造一个对象*/
        /* User user = (User) aClass.newInstance();*//*本质是调用了类的无参构造器*//*
        System.out.println(user);
        *//*通过构造器创建对象*/
        /*Constructor constructor = aClass.getDeclaredConstructor(String.class, int.class, int.class);
        User user1 = (User) constructor.newInstance("张三", 001, 18);
        System.out.println(user1);*/
        /*通过反射调用方法*/
        User user3 = (User) aClass.newInstance();
        /*通过反射获取一个方法*/
        Method setName = aClass.getDeclaredMethod("setName", String.class);
        setName.invoke(user3, "李四");/*invoke激活的意思*/
        System.out.println(user3.getName());
        /*通过反射操作属性*/
        User user4 = (User) aClass.newInstance();
        Field name = aClass.getDeclaredField("name");
        /*不能直接操作私有属性,我们需要关闭程序的安全检测,属性或者方法的setAccessible(true);*/
        name.setAccessible(true);
        name.set(user4, "王五");
        System.out.println(user4.getName());
        System.out.println("-------------------------结束-------------------------");

    }
}

*setAccessible *

  Method和Field\Constructor对象都有setAccessible()方法。
  setAccessible作用是启动和禁用访问安全检查的开关。
  参数值为true则指示反射的对象在使用时应该取消Java语言访问检查:提高反射的效率。如果代码中必须使用反射,这句代码被频繁使用,那么请设置为true,即可接触;使得原本无法访问的私有成员也可以访问。
  参数值为false(默认值)则指示反射的对象应该实施Java语言访问检查。

package com.ly.reflection;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/*@SuppressWarnings("all")*/
/*分析性能问题*/
public class Test10 {
    /*反射方法调用*/

    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
        System.out.println("-------------------------开始-------------------------");
        test01();
        test02();
        test03();
        System.out.println("-------------------------结束-------------------------");
    }

    /*普通方式调用*/
    public static void test01() {
        User user = new User();
        for (int i = 0; i < 1000000000; i++) {
            user.getName();
        }
        long startTime = System.currentTimeMillis();
        long endTime = System.currentTimeMillis();
        System.out.println("普通方法调用10亿次:" + (endTime - startTime) + "ms");
    }

    /*反射方法调用 关闭检测*/
    public static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class aClass = user.getClass();
        Method getName = aClass.getDeclaredMethod("getName", null);
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user, null);
        }
        long startTime = System.currentTimeMillis();
        long endTime = System.currentTimeMillis();
        System.out.println("反射方法调用10亿次:" + (endTime - startTime) + "ms");
    }

    /*反射方法调用 关闭检测*/
    public static void test03() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class aClass = user.getClass();
        Method getName = aClass.getDeclaredMethod("getName", null);
        getName.setAccessible(true);
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user, null);
        }
        long startTime = System.currentTimeMillis();
        long endTime = System.currentTimeMillis();
        System.out.println("关闭检测,反射方法调用10亿次:" + (endTime - startTime) + "ms");
    }
}

-------------------------开始-------------------------
普通方法调用10亿次:0ms
反射方法调用10亿次:0ms
关闭检测,反射方法调用10亿次:0ms
-------------------------结束-------------------------

反射操作泛型

  通过反射去操作泛型,Java里面的泛型是一种约束机制,保证代码的安全性和免去一些强制转换的问题,但是编译成功之后,所有和泛型有关的就会被擦除。
  为了操作这些泛型,Java新增了ParamrterizedType、GenericArrayType、TypeVariable和WildcardType几种类型代表不能被归一到Class类型,并且和原类型齐名。

package com.ly.reflection;

import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Map;

/*@SuppressWarnings("all")*/
/*通过反射获取泛型*/
public class Test11 {
    public static void main(String[] args) throws NoSuchMethodException {
        System.out.println("-------------------------开始-------------------------");
        Method method = Test11.class.getMethod("test01", Map.class, List.class);
        Type[] genericParameterTypes = method.getGenericParameterTypes();/*getGenericParameterTypes获得方法的参数类型*/
        for (Type genericParameterType : genericParameterTypes) {
            /*genericParameterType的参数类型是否等于ParameterizedType结构化参数化类型*/
            if (genericParameterType instanceof ParameterizedType) {
                /*如果ParameterizedType是一个参数类型,所以我们将它强转出来,之后就可以调用getActualTypeArguments方法获得真是参数类型*/
                Type[] actualTypeArguments = ((ParameterizedType) genericParameterType).getActualTypeArguments();
                for (Type actualTypeArgument : actualTypeArguments) {
                    System.out.println(actualTypeArgument);
                }
            }
        }

        method = Test11.class.getMethod("test02", null);
        Type genericReturnType = method.getGenericReturnType();
        if (genericReturnType instanceof ParameterizedType) {
            /*如果ParameterizedType是一个参数类型,所以我们将它强转出来,之后就可以调用getActualTypeArguments方法获得真是参数类型*/
            Type[] actualTypeArguments = ((ParameterizedType) genericReturnType).getActualTypeArguments();
            for (Type actualTypeArgument : actualTypeArguments) {
                System.out.println(actualTypeArgument);
            }
        }
        System.out.println("-------------------------结束-------------------------");
    }

    public void test01(Map<String, User> map, List<User> list) {
        System.out.println("test01");
    }

    public Map<String, User> test02() {
        System.out.println("test02");
        return null;
    }
}

-------------------------开始-------------------------
class java.lang.String
class com.ly.reflection.User
class com.ly.reflection.User
class java.lang.String
class com.ly.reflection.User
-------------------------结束-------------------------

什么是ORM

  Object relation Mapping —> 对象关系映射
  类和表结构对应;属性和字段对应;对象和记录对应

package com.ly.reflection;

import java.lang.annotation.*;
import java.lang.reflect.Field;

/*类名的注解*/
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@interface Tablekuang {
    String value();
}

/*属性的注解*/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface Fieldkuang {
    String columnName();

    String type();

    int length();
}

/*@SuppressWarnings("all")*/
/*练习反射操作注解*/
public class Test12 {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
        System.out.println("-------------------------开始-------------------------");
        Class aClass = Class.forName("com.ly.reflection.Studeng2");
        /*通过反射获得注解*/
        Annotation[] annotations = aClass.getAnnotations();
        for (Annotation annotation : annotations) {
            System.out.println(annotation);
        }
        /*获得注解的vaule的值*/
        /*通过aClass.getAnnotation获得指定的注解,生成相应的注解信息,把这个信息value出来*/
        Tablekuang tablekuang = (Tablekuang) aClass.getAnnotation(Tablekuang.class);
        String value = tablekuang.value();
        System.out.println(value);
        /*获得类指定的注解*/
        Field f = aClass.getDeclaredField("name");/*获得这个name类*/
        Fieldkuang annotation = f.getAnnotation(Fieldkuang.class);/*获取这个指定类的注解*/
        System.out.println(annotation.columnName());/*得到annotation分别打印相应的值*/
        System.out.println(annotation.type());
        System.out.println(annotation.length());
        System.out.println("-------------------------结束-------------------------");
    }
}

@Tablekuang("db_student")
class Studeng2 {
    @Fieldkuang(columnName = "db_id", type = "int", length = 10)
    private int id;
    @Fieldkuang(columnName = "db_age", type = "int", length = 10)
    private int age;
    @Fieldkuang(columnName = "db_name", type = "varchar", length = 3)
    private String name;

    public Studeng2() {
    }

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

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

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public int getAge() {
        return age;
    }

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

    public String getName() {
        return name;
    }

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

-------------------------开始-------------------------
@com.ly.reflection.Tablekuang(value=db_student)
db_student
db_name
varchar
3
-------------------------结束-------------------------

总结

  Annotation的作用:可以被其他程序读取的功能。
  内置注解:@Override、@Deprecated、@SuppressWarnings(抑制编译的警告信息)。
  使用@interface去自定义一个注解。
  关于自定义注解中的源注解:@Target:用于描述注解的使用范围;@Retention:表示在什么级别有效。
  反射相关的主要API:
    java.lang.Class:代表一个类;
    java.lang.reflect.Method:代表类的方法;
    java.lang.reflect.Field:代表类的成员变量;
    java.lang.reflect.Constructor:代表类的构造器。
  无论有多少对象,它创建出来的类只有一个Class
  引导类加载器:Bootstap Classloader;扩展类加载器:Extension Classloader;系统类加载器:System Classloader。
  通过反射,调用类中的方法,通过Method类完成:1.通过Class类的getMethod(String name,Class…parameterTypes)方法取得一个Method对象,并设置次方法操作时所需要的参数类型;2.之后使用(Object obj,Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

游荡在雾中的霾

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

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

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

打赏作者

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

抵扣说明:

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

余额充值