#拓展开题:
类的加载与ClassLoader的理解

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

类加载器的作用
 
动态创建对象执行方法
有了Class对象,能做什么?

newInstance()是一个Java中使用反射创建对象实例的方法,通过该方法可以获取类的构造函数,并创建出一个新的对象实例。(类似于new)
method.invoke()就是调用类中的方法,第一个参数是obj(对象),在我们平常使用过程中用到的是类,类是对象的一个集合,第二个参数是args(参数),是调用invoke这个方法所使用的参数,我们使用是一般是类中的方法(method),因此invoke方法我们也可以这样理解:invoke(class,method),相当于把类中的方法参数化了。(简单理解:激活的意思)
field.setAccessible是Java反射中的一个方法,用于设置一个字段(Field)的可访问性。通过调用该方法,可以使得一个私有字段也可以被访问和修改。具体来说,该方法接受一个布尔值作为参数,如果为true,则表示该字段可以被访问和修改;如果为false,则表示该字段仍然是私有的,无法被访问和修改。
package reflection;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Test04 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //获得Class对象
        Class c1 = Class.forName("reflection.User");
        //构造一个对象
        User user = (User) c1.newInstance();//本质是调用了类的无参构造
        System.out.println(user);
        //通过构造器创建对象
        Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user2 = (User) constructor.newInstance("权业轩", 001, 16);
        System.out.println(user2);
        //通过反射调用普通方法
        User user3 = (User)c1.newInstance();
        //通过反射获取一个方法
        Method setName = c1.getDeclaredMethod("setName", String.class);
        setName.invoke(user3,"龙阳");//invoke:激活的意思
        System.out.println(user3.getName());
        //通过反射操作属性
        User user4 = (User)c1.newInstance();
        Field name = c1.getDeclaredField("name");
        name.setAccessible(true);//取消安全检测(因为上面权限不够,name的权限是private)
        name.set(user4,"龙葵");
        System.out.println(user4.getName());
    }
}

补充:
调用指定的方法:

 

性能对比分析:
package reflection;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
//分析性能问题
public class Test05 {
    //普通方式调用
    public static void test01() {
        User user = new User();
        long statrTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            user.getName();
        }
        long endTime = System.currentTimeMillis();
        System.out.println("普通方式执行10亿次:" + (endTime - statrTime) + "毫秒");
    }
    //反射方法调用,无setAccessible
    public static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();
        Method getName = c1.getDeclaredMethod("getName", null);
        long statrTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user, null);
        }
        long endTime = System.currentTimeMillis();
        System.out.println("反射方式执行10亿次:" + (endTime - statrTime) + "毫秒");
    }
    //反射方法调用,关闭检测
    public static void test03() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();
        Method getName = c1.getDeclaredMethod("getName", null);
        getName.setAccessible(true);
        long statrTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user, null);
        }
        long endTime = System.currentTimeMillis();
        System.out.println("反射方式关闭检测执行10亿次:" + (endTime - statrTime) + "毫秒");
    }
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
        test01();
        test02();
        test03();
    }
}
运行结果:

由于输出关系,反射方式执行10亿次应该是4.057毫秒,关闭检测后是1.846毫秒,所以建议关闭检测来执行反射。



















