反射详解

反射

一、什么是反射:
能够分析类能力的程序称为反射(reflective)。
二、获得class类对象的三种方式

在程序运行期间,Java 运行时系统始终为所有的对象维护一个被称为运行时的类型标识。这个信息跟踪着每个对象所属的类。 虚拟机利用运行时类型信息选择相应的方法执行。
然而,可以通过专门的 Java 类访问这些信息。保存这些信息的类被称为 Class, 这个名字很容易让人混淆。Object 类中的 getClass( ) 方法将会返回一个 Class 类型的实例。

  • Student.class 类名.class

  • Student stu = new Student();

    stu.getClass(); 对象.getClass();

  • Class.forName(“com.qcby.entity.Student”); 字符串是一个全限定名 包名.类名的形式

创建对象

//等价的
Student student = Student.class.newInstance();
Student student1 = new Student();

通过反射只能拿到参数类型,拿不到参数名

可以拿到域的名字

 public class Test {
     private int age;
     @Override
     public String toString() {
         return "Test{" +
                 "age=" + age +
                 '}';
     }
     public static void main(String[] args) {
         Test test = new Test();
         try {
             Field age = Test.class.getDeclaredField("age");
             //设置可访问非常重要
             age.setAccessible(true);
             age.set(test,20);
             System.out.println(test);
             System.out.println(age.get(test));
         } catch (NoSuchFieldException e) {
             e.printStackTrace();
         } catch (IllegalAccessException e) {
             e.printStackTrace();
         }
     }
 }

1、我们通过第一个try-catch可以看到即使是对象中的私有属性通过反射也是可以访问到的,前提是设置可访问的属性。
2、通过第二个try-catch的应用我们便实现了利用反射机制中的invoke方法来调用对象中的方法,也就是我们通常所说的代理,从这里我们可以看到,反射和代理是在一起使用的。

Method toString = Test.class.getDeclaredMethod("toString");
Object invoke = toString.invoke(test);
System.out.println(invoke);
import com.sun.org.apache.xpath.internal.operations.Mod;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Scanner;

public class RefelectionTest {
    public static void main(String[] args) {
        String name;
        if (args.length>0 ) name = args[0];
        else {
            Scanner in = new Scanner(System.in);
            System.out.println("Enter class name(e.g.java.util.Date):");
            name = in.next();
        }
        try {
            Class cl = Class.forName(name);
            Class supercl = cl.getSuperclass();
            String modifiers = Modifier.toString(cl.getModifiers());
            if(modifiers.length()>0){
                System.out.print(modifiers+" ");
            }
            System.out.print("class"+name);
            if (supercl!=null && supercl!=Object.class) System.out.print("extends"+supercl.getName());
            System.out.print("\n{\n");
            printConstructors(cl);
            System.out.println();
            printMethods(cl);
            System.out.println();
            printFields(cl);
            System.out.println("}");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        System.exit(0);
    }



    private static void printConstructors(Class cl) {
        Constructor[] constructors = cl.getDeclaredConstructors();
        for (Constructor c:constructors) {
            String name = c.getName();
            System.out.print("   ");
            String modifiers = Modifier.toString(c.getModifiers());
            if (modifiers.length()>0) System.out.print(modifiers+" ");
            System.out.printf(name+"(");

            Class[] paramTypes = c.getParameterTypes();
            for (int i = 0; i < paramTypes.length; i++) {
                if (i>0) System.out.print(",");
                System.out.printf(paramTypes[i].getName());
            }
            System.out.println(");");
        }
    }
    private static void printMethods(Class cl) {
        Method[] methods = cl.getDeclaredMethods();
        for (Method m:methods) {
            Class retType = m.getReturnType();
            String name = m.getName();
            System.out.printf("  ");
            String modifiers = Modifier.toString(m.getModifiers());
            if (modifiers.length()>0) System.out.printf(modifiers+" ");
            System.out.printf(retType.getName()+" "+name+"(");

            Class[] paramTypes = m.getParameterTypes();
            for (int i = 0; i < paramTypes.length; i++) {
                if (i>0) System.out.print(",");
                System.out.printf(paramTypes[i].getName());
            }
            System.out.println(");");
        }
    }
    private static void printFields(Class cl) {
        Field[] fields =cl.getDeclaredFields();
        for (Field f:fields) {
            Class type = f.getType();
            String name = f.getName();
            System.out.printf("  ");
            String modifiers = Modifier.toString(f.getModifiers());
            if (modifiers.length()>0) System.out.printf(modifiers+" ");
            System.out.println(type.getName() + " "+ name +";");
        }
    }

}

反射的具体用途

通过以上的介绍我们可以总结:

1、通过反射获取类信息;
2、根据类信息生成实例(实例化功能);
3、反射最终的目的是拿到类的实例。

反射补充

反射的性能非常慢,是毫秒级的消耗。

所有的文件类型都是储存在磁盘上的

反射设置可以访问之后可以访问到私有属性

当程序没有主方法的时候,程序是不能自己独立运行的。此时它只能依靠别的程序来运行。比如servlet,它自己是没办法运行的,它需要依靠tomcat来运行

tomcat中依靠注解来分辨哪个是需要实例化的servlet类信息

反射的应用

一般反射在框架中应用是比较多的,但是为什么要在框架中运用反射机制呢?用new关键字来创建对象不是更好的选择吗?下面给出解答

例:tomcat是个容器,容器的开发者不知道项目的开发者会写什么类,所以无法用new的方法来给类进行实例化,所以容器开发者也会写出一个类的模板(servlet)来告诉项目的开发者开发规则是什么,当容器运行的时候,用反射的机制按照class.forName()这个方式获取类的信息然后在newInstance()进行实例化然后再运行,容器的开发者会把这个类的模板提供给项目开发者,项目开发者需要根据这个模板的规则来写对应的类,如果没有根据规则来写,也是不可以运行的。

那么,框架如果用new的方式来创建对象的话,首先需要有对应的类,也就是类是写死的,没有办法根据项目的开发进行动态的改变,而这就失去了开发框架的初衷

框架使用new关键字和反射机制的图示:
图一是写死的,图二是用反射机制
在这里插入图片描述
如上就是对框架运用反射机制的解释。学习快乐!加油!

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Python中,反射是指通过字符串的形式去访问对象的属性、方法和其他成员。Python中的每个对象都有一组内置的特殊属性和方法,比如 `__dict__`、`__class__`、`__name__`、`__bases__` 等等。这些属性和方法可以通过反射机制来访问和操作。 Python中的反射通常涉及到以下三个内置函数: 1. `hasattr(object, attribute)`:检查对象是否具有指定的属性或方法。 2. `getattr(object, attribute[, default])`:获取对象的属性或方法。如果属性或方法不存在,则返回默认值(如果指定了)或引发 AttributeError 异常。 3. `setattr(object, attribute, value)`:设置对象的属性或方法。 下面是一些示例: ```python class MyClass: def __init__(self): self.x = 10 def my_method(self): print("Hello, World!") # 创建一个对象 obj = MyClass() # 使用 getattr 获取对象的属性和方法 print(getattr(obj, 'x')) # 输出: 10 getattr(obj, 'my_method')() # 输出: Hello, World! # 使用 hasattr 检查对象是否具有属性和方法 print(hasattr(obj, 'x')) # 输出: True print(hasattr(obj, 'y')) # 输出: False # 使用 setattr 设置对象的属性和方法 setattr(obj, 'y', 20) print(obj.y) # 输出: 20 setattr(obj, 'my_method', lambda: print("Hello, Python!")) obj.my_method() # 输出: Hello, Python! ``` 除了使用内置函数之外,Python中还有另一种反射机制,即使用元类。元类是用于创建类的类,可以通过元类来动态地创建、修改和删除类的属性和方法。元类的使用超出了本文的范围,感兴趣的读者可以自行学习。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值