java 反射

一、获取Class实例的三种方式

Ps:Fool f=new  Fool();
Class c1=f.getClass;    //已知类的名称
Class c2=Fool.class;    //已知类的实例对象
Class c3=Class.forName("com.aa.bb.Fool");   //已知类的全限定名

二、静态加载类与动态加载类

1、静态加载类(编译时,加载的类是静态加载类)
Foo foo = new Foo();
new创建对象是静态加载类,编译时就要加载所有要用到的类,如果其中一个类有问题,就无法通过编译。

2、动态加载类(运行时,加载的类是动态加载类)
Class.forName("com.reflect.Foo");
编译时不检查用到的类是否存在,运行时再检查。

三、常用方法

public class TestReflect {    
    public static void printClassmessage(Object obj){
        Class c=obj.getClass();   
        try {
            //通过参数获得指定的构造方法
            Constructor ctr= c.getConstructor(int.class,String.class);
            //创建一个对象
            ctr.newInstance(999,"小米");  
        } catch (Exception e) {
            e.printStackTrace();
        }

        //获取该类的所有public构造函数;
        Constructor[] ct = c.getConstructors();

        //获取该类自行声明的所有构造函数;
        Constructor[] cts = c.getDeclaredConstructors();


        try {
            //通过方法名和参数获得指定的方法(两种写法)
            Method m1 = c.getMethod("eat",new Class[]{int.class,String.class});
            Method m2 = c.getMethod("eat",int.class,String.class);

            //如果无参
            Method m3 = c.getMethod("eat",new Class[]{});
            Method m4 = c.getMethod("eat");

           //执行一个方法,第一个参数是执行方法的对象,第二个参数是方法所需要的参数
            m1.invoke(obj,1,"1");
            m2.invoke(obj,new Object[]{1,"1"});
        } catch (Exception e) {
            e.printStackTrace();
        }



        //获取该类的所有public方法,包括从父类继承的方法;
        Method[] ms=c.getMethods();
        //获取该类自行声明的所有方法,不论访问权限;
        Method[] mes=c.getDeclaredMethods();

        //获取该类的所有public成员变量
        Field[] fl=c.getFields();
        //获取该类自行声明的所有成员变量,不包括从父类继承来的
        Field[] fld=c.getDeclaredFields();

        for (int i=0;i<ms.length;i++){
            //获取方法的返回值类类型,如:int.class
            Class returnType=ms[i].getReturnType();
            //获取方法的返回值类型的名字
            returnType.getName();

            //获取方法名
            ms[i].getName();

            //获得参数列表类型的类类型,如:int.class
            Class[] paramTypes = ms[i].getParameterTypes();
            for (Class cl:paramTypes){
                System.out.println(cl.getName());
            }

        }

        for (Field field:fl){
            //获取成员变量类型的类类型
            Class fieldType = field.getType();
            fieldType.getName();

            //获取成员变量的名称
            String fieldName = field.getName();
        }


        for (Constructor constructor:ct){
            //获取构造函数的名称
            constructor.getName();

            //获取构造函数的参数列表
            Class[] paramTypes = constructor.getParameterTypes();
            for (Class cz:paramTypes){
                //获取构造函数的参数名称
                cz.getName();
            }
        }
    }  
}
### Java反射机制使用教程及常见问题 Java反射机制是一种强大的工具,允许程序在运行时动态地检查和操作类、接口、字段和方法等内部信息[^1]。以下是关于Java反射机制的详细教程以及一些常见的问题。 #### 1. 反射机制的基本概念 Java反射机制是Java语言中一种动态访问、检测和修改自身的能力,主要作用是在运行时获取类的完整结构信息并对其进行操作[^2]。通过反射,可以实现以下功能: - 动态创建对象。 - 动态调用方法。 - 动态访问字段。 - 动态处理注解。 #### 2. 核心类与API Java反射的核心类位于`java.lang.reflect`包中,主要包括以下几类: - `Class`:表示类的运行时类对象,可以通过`Class.forName(String className)`或`Object.getClass()`获取。 - `Field`:表示类的成员变量。 - `Method`:表示类的方法。 - `Constructor`:表示类的构造方法。 #### 3. 使用示例 以下是一个简单的反射机制使用示例,展示了如何通过反射创建对象并调用方法。 ```java import java.lang.reflect.Method; public class ReflectionExample { public static void main(String[] args) throws Exception { // 获取目标类的Class对象 Class<?> cls = Class.forName("com.example.MyClass"); // 创建目标类的实例 Object obj = cls.getDeclaredConstructor().newInstance(); // 获取目标类的方法 Method method = cls.getMethod("myMethod", String.class); // 调用目标类的方法 method.invoke(obj, "Hello Reflection"); } } ``` #### 4. 性能问题 反射机制虽然强大,但在性能上存在一定的开销。例如,在大量调用反射方法时,其性能可能显著低于直接调用[^4]。因此,在实际开发中应权衡使用反射的必要性。 #### 5. 常见问题 - **安全性问题**:反射可以访问私有成员,这可能导致破坏封装性[^1]。 - **性能问题**:反射调用方法的性能比普通方法调用低[^4]。 - **异常处理**:反射操作容易抛出多种异常,如`ClassNotFoundException`、`NoSuchMethodException`等,需要妥善处理。 #### 6. 实际应用场景 Java反射机制在许多实际场景中发挥了重要作用,包括但不限于框架设计、动态代理、依赖注入、测试工具和序列化/反序列化等[^3]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值