java 反射

Reflection 是 Java 程序开发语言的特征之一,它允许运行中的 Java 程序对自身进行检查,或者说“自审”,并能直接操作程序的内部属性和方法。

 

Java类中的反射就是把类中的属性和方法映射成相应的类.

 

Java的类反射所需要的类,它们分别是:Field、Constructor、Method、Class.

 

反射的三种实现方法:

 

1. 类名.class

2.对象名.getClass()

3.Class.forName("类的完整名称")

 

Class.forName的作用,如果虚拟机内存中不存在该类的字节码则由类加载器载,虚拟机内存中存在就直接返回。一个类的字节码在内存中只有唯一的一份。

 

取的字节码的三种方式

 

String.class;   
String.getClass();   
Class.forName("java.lang.String");  

 

1.通过反射取的String类的实例对象

 

import java.lang.reflect.Constructor;   
public class ReflectDemo01 {   
       
    public static void main(String args[]) throws Exception{   
           
        Constructor strConstructor = String.class.getConstructor(String.class);   
           
        String str = (String)strConstructor.newInstance(new Object[]{"123"});   
           
        System.out.println(str);   
    }   
}  

 

2.通过反射调用String类的charAt(int index)方法 

 

import java.lang.reflect.Method;   
  
public class ReflectDemo01 {   
       
    public static void main(String args[]) throws Exception{   
  
           
        String str = "TestReflect";   
           
        //Method charAt = str.getClass().getMethod("charAt",int.class);   
  
        Method charAt = String.class.getMethod("charAt",int.class);   
           
        System.out.println(charAt.invoke(str,0));   
    }   
}

 

3.通过反射调用String类的静态方法valueOf(boolean b);

 

import java.lang.reflect.Method;   
  
public class ReflectDemo02 {   
    public static void main(String args[]) throws Exception{   
           
           
        Method valueOf = String.class.getMethod("valueOf",boolean.class);   
           
                                 /**调用时传null,因为静态方法不属于对象**/  
        System.out.println(valueOf.invoke(null,true));   
           
           
               
    }   
}

 

4.用反射返回类的属性

 

import java.lang.reflect.Field;   
  
    
public class ReflectDemo03 {   
       
    public static void main(String args[]) {   
       
        /**  
            返回类所声明的属性(成员变量),    包括私有属性  
        **/  
           
        Field[] allFields = Student.class.getDeclaredFields();   
           
        for(Field field:allFields){   
            if(field.getType() == String.class){       
                System.out.println(field.getGenericType());    
            }   
               
            if(field.getType() == Integer.TYPE){   
                System.out.println(field.getGenericType());            
            }   
        }    
           
        /**  
            返回类所声明可访问的公共属性    
          
        **/    
           
        Field[] pubFields = Student.class.getFields();   
           
        for(Field field:pubFields){   
            if(field.getType() == String.class){       
                System.out.println(field.getGenericType());    
            }   
               
            if(field.getType() == Integer.TYPE){   
                System.out.println(field.getGenericType());            
            }   
        }   
    }      
}   
  
class Student {   
    public String name;   
    private int age;           
}

 

5.通过反射修改类的私有属性

 

import java.lang.reflect.Field;   
  
public class ReflectDemo04 {   
       
    public  static void main(String[] args) throws Exception{   
                   
        Student student = new Student("小华",16);   
           
        System.out.println(student);   
         /**  
            返回类中所有声明的属性,目前只能让你看到,不能够使用.  
          
            也可以通过getDeclaredField(String name)查看某一个属性的声明(包括私有属性).   
 
        **/  
  
        Field []prvFields = student.getClass().getDeclaredFields();   
           
        for(Field prvField:prvFields) {   
               
               
            if(prvField.getType() == String.class){   
                   
                /**  
                    取消Java 语言默认的访问控制检查  
                **/  
                   
                prvField.setAccessible(true);   
                       
                prvField.set(student,"小猪");    
            }   
               
            if(prvField.getType() == Integer.TYPE) {   
                   
                /**  
                    取消Java 语言默认的访问控制检查  
                **/  
                   
                prvField.setAccessible(true);   
                       
                prvField.setInt(student,18);           
            }   
        }   
           
        System.out.println(student);           
    }   
}   
  
class Student {   
       
    private String name;   
    private int age;       
       
    public Student(String name,int age){   
        this.name = name;   
        this.age = age;    
    }   
       
    @Override  
    public String toString() {   
        return "姓名:"+this.name + "  年龄:"+this.age;     
    }      
}

 

 

### 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、付费专栏及课程。

余额充值