java的反射

声明:部分资料来源于网络,仅供本人学习

1.*****简述一下反射

反射机制是通过操作类的字节码即.class文件,对获得类的所有信息。比如可获得类中的注解,类的父类superClass,实现的接口interface,属性Field,构造方法Constructor,方法Mehod等。

2.*****获得类的字节码Class的三种方式

Class.forName("全限定类名")

对象.getClass() 如"abc".getClass()返回 java.lang.String

数据类型.class 如int.class 返回 int ,类名.class,基本数据类型.class....

3.*****Class.forName("全限定类名")执行时,会加载该类的字节码文件到方法区,执行该类的静态代码块。

静态代码块是在字节码加载的时候执行,只执行一次。可用于加载数据库驱动等作用。

4.通过反射实例化对象

  Class name = Class.forName("java.lang.String");
  //该方法已弃用
  //调用无参构造,若类中没有无参构造会报错
  Object obj=name.newInstance();

5.*****读取类路径下的文件的两种方法(linux通用)

//相对路径是以src为起点(根),只能写src包下的文件
//第一种方法 获得绝对路径
//getResource("")方法中参数是以类路径为起点的相对路径。 
String path = Thread.currentThread().getContextClassLoader().getResource("类路径下的文件").getPath();
FileReader reader=new FileReader(path);


//第二种方法 获得流
InputStream stream = Thread.currentThread().getContextClassLoader().getResourceAsStream("类路径下的文件");

maven项目的项目结构

|src

----main

----------java

----------------自定义包与类 如com.zhaojisu.XxApplication

----------resource

----------------------static

----------------------tmplates

----------------------application.properties

----------------------等等...   

----test

|target

|pom.xml

编译后会将java包和resource内的包放在类路径下,而src manin test java rource在编译后都不存在。以下为例子。

 

6.读取配置文件配合反射创建对象

//读取配置文件
String path=Thread.currentThread().getContextClassLoader().getResource("src下的文件,填相对路径").getPath();

FileReader reader=new FileReader(path);

Properties properties=new Properties();
//配置文件加keyvalue记载到properties集合中
properties.load(reader);

String o = (String)properties.get("className");//className=java.lang.String

Class aClass = Class.forName(o);

String instance = (String)aClass.newInstance();
InputStream stream = Thread.currentThread().getContextClassLoader().getResourceAsStream("类路径下的文件");
Properties properties=new Properties();
properties.load(stream);
String o = (String)properties.get("className");//className=java.lang.String
Class aClass = Class.forName(o);
String instance = (String)aClass.newInstance();
System.out.println(instance);

使用反射创建对象非常灵活,不需要在代码中写明要创建的类型,只需要在配置文件中确定,这样的代码具有通用性,不需要修改java代码就能创建任意类的实例对象。

7.读取配置文件更厉害的方法 资源绑定器ResourceBundle

//getBundle方法的参数是类路径下的文件。只能是.properties文件
//不能加上.properties
ResourceBundle resourceBundle=ResourceBundle.getBundle("static/test1");

String className = (String)resourceBundle.getObject("className");

System.out.println(className);

8.操作Field

        Class aClass = Class.forName("java.lang.String");
        //获得所有公共属性
        Field[] fields = aClass.getFields();
        //获得所有属性 包括私有
        Field[] fields2 = aClass.getDeclaredFields();
        for (Field field : fields2) {
            //获得属性的修饰符
            int modifiers = field.getModifiers();
            //转成数字对应的修饰符字符串
            Modifier.toString(modifiers);
            //获得属性的类型的字节码
            Class type = field.getType();
            //获得属性的名字
            String name = field.getName();
        }

通过反射设置对象属性

9.操作Method

        Class<?> aClass = Class.forName("java.lang.Integer");
        //获得所有公共方法包括继承得到的
        aClass.getMethods();
        //获得本类中声明的全部方法 包括私有方法
        Method[] methods = aClass.getDeclaredMethods();
        for (Method method : methods) {
            //方法的修饰符
            String s = Modifier.toString(method.getModifiers());
            //方法的返回值
            Class<?> returnType = method.getReturnType();
            //方法的参数类型列表
            Class<?>[] parameterTypes = method.getParameterTypes();
        }

通过反射调用对象的方法

 10.操作Constructor

        Class<?> aClass = Class.forName("java.lang.Integer");
        //获得类中声明的全部构造方法
        Constructor<?>[] constructors = aClass.getDeclaredConstructors();
        for (Constructor<?> constructor : constructors) {
            //构造方法名
            String name = constructor.getName();
            //构造方法的参数的字节码
            Class<?>[] parameterTypes = constructor.getParameterTypes();
        }
    }     

通过反射调用构造方法创建对象

        Class<?> aClass = Class.forName("java.io.File");
        //获得类中声明的全部构造方法
        try {
            Constructor<?> constructor = 
                   aClass.getDeclaredConstructor(String.class,int.class);
            constructor.newInstance("aaa",1);
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }

11.获得父类父接口

        Class aClass = Class.forName("java.io.File");
        Class[] interfaces = aClass.getInterfaces();
        Class superclass = aClass.getSuperclass();

### 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]。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值