动态创建对象执行方法

创建类的对象:调用Class对象的newInstance()方法

  • 类必须有一个无参数的构造器
  • 类的构造器的访问权限需要足够

思考?难道没有无参的构造器就不能创建对象了吗?

答:只要在操作的时候明确的调用类中的构造器,并将参数传递进去之后,才可以实例化操作

步骤如下:

  • 通过Class类的getDeclaredConstructor(Class...parameterTypes)取得本类的指定形参类型的构造器
  • 向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数
  • 通过Constructor实例化对象

调用指定的方法:

通过反射,调用类中的方法,通过Method类完成

  • 通过Class类的getMethod(String name,Class...parameterTypes)方法取得一个Method对象,并设置此方法操作时所需要的参数类型
  • 之后使用Object invoke(Object obj,Object[] args)进行调用,并向方法中传递要设置的obj对象的学习

对于Object invoke(Object obj,Object[] args)

  • Object对应原方法的返回值,若原方法无返回值,此时返回null
  • 若原方法为静态方法,此时Object obj可为null
  • 若原方法形参列表为空,则Object[] args为空
  • 若原方法声明为private,则需要调用此invoke()方法前,显示调用方法对象的setAccessible(true)方法,将可访问private的方法

setAccessible(true)

  • Method和Field、Constructor对象都有setAccessible()方法
  • setAccessible()的作用时启动和禁用访问安全检查的开关
  • 参数值为true则指示反射的对象再使用时应该取消Java语言访问检查
  • 提高反射的效率。如果代码中必须用到反射,而该句代码需要频繁的被调用,那么请设置为true
  • 使得原本无法访问的私有成员也可以访问
  • 参数值为false则指示反射的对象应该实施Java语言访问检查
package com.cbbpp.reflection;

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

public class test08 {
    public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {
        //获得Class对象
        Class c1 = Class.forName("com.cbbpp.reflection.User");
        //通过无参构造直接Class对象.newInstance()构造一个对象
        //本质上是调用了类的无参构造器,若类中生删除无参构造器,这这种形式生成对象会报错
        User user1 = (User) c1.newInstance();
        System.out.println(user1);
        //不通过无参构造器,而通过构造器实例化拿到constructor,再使用constructor.newInstance()实例化对象
        Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);
        User user2 = (User) constructor.newInstance("土拨鼠", 1, 19);
        System.out.println(user2);
        System.out.println("=============================================================");
        //通过反射调用普通方法
        Class c2 = Class.forName("com.cbbpp.reflection.User");
        User user3 = (User) c2.newInstance();
        //通过反射获取一个方法
        Method setName = c2.getDeclaredMethod("setName", String.class);
        //invoke  :  激活的意思
        //invoke(对象,"args")
        setName.invoke(user3, "土老爷");
        System.out.println(user3.getName());
        System.out.println("=============================================================");
        //通过反射操作属性
        User user4 = (User) c2.newInstance();
        Field name = c2.getDeclaredField("name");
        //不能直接操作私有属性,需要通过关闭程序的安全监测,属性或者方法的.setAccessible()设置为true
        name.setAccessible(true);
        name.set(user4,"老周");
        System.out.println(user4.getName());
    }
}

//结果
User{name='null', id=0, age=0}
User{name='土拨鼠', id=1, age=19}
=============================================================
土老爷
=============================================================
老周

Process finished with exit code 0

三种方式(普通方式、反射方式、关闭检测反射反射)性能检查

package com.cbbpp.reflection;

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

public class Test09 {
    //分析性能问题
    //普通方式
    public static void test01(){
        User user = new User();
        long startTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            user.getName();
        }
        long endTime = System.currentTimeMillis();
        System.out.println("普通方式执行10亿次:"+(endTime-startTime)+"ms");
    }
    //反射方式
    public static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        User user = new User();
        Class c1 = user.getClass();
        Method getName = c1.getDeclaredMethod("getName", null);
        long startTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user,null);
        }
        long endTime = System.currentTimeMillis();
        System.out.println("反射方式执行10亿次:"+(endTime-startTime)+"ms");
    }
    //关闭检查反射方式
    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 startTime = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            getName.invoke(user,null);
        }
        long endTime = System.currentTimeMillis();
        System.out.println("关闭检查反射方式执行10亿次:"+(endTime-startTime)+"ms");
    }
    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException {
        test01();
        test02();
        test03();
    }
}

结果:

普通方式执行10亿次:3ms
反射方式执行10亿次:2503ms
关闭检查反射方式执行10亿次:1196ms

Process finished with exit code 0

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值