public class Student{
private int id=0;
public int getId(){
return id;
}
}
问题:能否将ReadOnlyClass 类的一个对象,把它的name属性的值由hello改为world?如果能,请写出实现代码。如果不能请说明理由。
答案:可以。利用java的反射
分析:任何一个类,我们可以得到它运行时的Class实例,对于Student类,我们可以通过Student.class得到它运行时的Class实例,接着我们可以通过该类的Class实例去获得这个id属性所对应的Field对象,我们知道对应一个类的属性都一个和它相关的Field对象存在,对于构造方法来说有一个Constructor对象存在,对于一个方法来说有一个对应的Method对象存在。通过这些我们可以利用反射来给这些属性动态的赋值。
首先我们看看JDK API中Class类的两个方法的描述:
public Field getField(String name)throws NoSuchFieldException,SecurityException
Returns a Field object that reflects the specified public member field of the class or interface represented by this Class object. The name parameter is a String specifying the simple name of the desired field.The field to be reflected is determined by the algorithm that follows. Let C be the class represented by this object:If C declares a public field with the name specified, that is the field to be reflected.If no field was found in step 1 above, this algorithm is applied recursively to each direct superinterface of C. The direct superinterfaces are searched in the order they were declared.If no field was found in steps 1 and 2 above, and C has a superclass S, then this algorithm is invoked recursively upon S. If C has no superclass, then a NoSuchFieldException is thrown.(翻译:返回一个Field对象,它返回此Class对象所表示的类或接口的指定public属性字段,id参数是一个int,用于指定所需字段的简称。要反映的字段由下面的算法确定。设C为次对象所表示的类:如果C声明一个带有指定名的公共字段,则它就是要反映的字段。如果在第1步中没有找到任何字段,则该算法被递归的应用与C的每一个直接超接口。直接超接口按其声明的顺序进行搜索,如果在第1,2两步没有找到任何字段,且C有一个超类S,则在S上递归调用该算法,如果C没有超类则抛出NoSuchFieldException.)
public Field getDeclaredField(String name) throws NoSuchFieldException,SecurityException
Returns a Field object that reflects the specified declared field of the class or interface represented by this Class object. The name parameter is a String that specifies the simple name of the desired field. Note that this method will not reflect the length field of an array class.(翻译:返回一个Field对象,该对象反映次Class对象所表示的类或接口的指定以声明字段,id参数是一个int,它指定所需字段的简称,次方法不反映数组类的length字段。)
从上面的JDK API我们知道,要得到id属性对应的Filed对象,我们只能调用class的getDeclaredField方法,因为getField方法只能得到一个类的public的属性对应的Field对象,而这里的name属性是private的,我们通过Class的getDeclaredField方法得到id属性对应的Field对象后,我们就可以调用Filed对象的set方法给id赋值。由于id是私有属性,我们要想知道我们到底改变了id的值没有,我们可以通过Filed类的父类的setAccessible(boolean flag)方法类限制java语言 的访问限制。对于setAccessible方法我们看JDK文档:
public void setAccessible(boolean flag)throws SecurityException
Set the accessible flag for this object to the indicated boolean value. A value of true indicates that the reflected object should suppress Java language access checking when it is used. A value of false indicates that the reflected object should enforce Java language access checks.First, if there is a security manager, its checkPermission method is called with a ReflectPermission("suppressAccessChecks") permission. A SecurityException is raised if flag is true but accessibility of this object may not be changed (for example, if this element object is a Constructor object for the class Class). (翻译:将此对象的 accessible标志设置为指示的布尔值,值为true则指示反射的对象在使用时应该取消java语言访问检查,值为False则指示反射的对象应该事实java语言访问检查。
首先,如果存在安全管理器,则在ReflectPermission("suppressAccessChecks"))权限下调用checkPermission方法。如果falg为true,并且不能更改次对象的可访问性(例如,如果次元素对象是Class类的Constructor对象),。则会引发SecurityException。如果次对象是java.lang.Class类的Constructor对象,并且flag为true,则会引发SecurityException
所以:
public class StudentRef{
public static void main(String[] args){
Student stu=new Student();
Class<?> cla=Student.class ;
Field filed = cla.getDeclaredField("id");
field.setAccessible(true);
field.set(stu,2);
System.out.println(stu.getId());
}
}
总结:对于一个类,它只有唯一个Class对象,它来标识这个对象,这个Class对象就能够获得这个类的结构上的特征。那那么通过Class对象就可以来获得这个类相应的构造方法,属性等。
获得某一个类它的Class对应有4种方法:
* 使用类的.class语法
* 使用类的对象的getClass()方法,getClass()方法在Object类里定义的
* 通过Class对象的forName()方法
* 对于包装类,可以通过.TYPE方法
通过类的反射机制,我们可以改变只读的私有(private)属性值
分享到:
相关推荐
标题“Java反射机制修改私有成员变量的实现_Reflection”着重讲述了如何利用反射来修改私有成员变量的值。在Java中,私有成员变量通常被视为封装的一部分,不被外部类直接访问。然而,通过反射,我们可以绕过这种...
Java反射机制是Java语言提供的一种强大功能,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在Java中,反射机制的核心类是`java.lang.Class`,它代表了类的信息。通过Class对象,我们可以...
3. **访问和修改私有属性和方法**:即使类的成员变量或方法被声明为private,反射机制也能访问和修改它们。 4. **动态调用方法**:可以调用任意一个对象的方法,无论其访问权限如何。 5. **生成动态代理**:用于实现...
本文将详细介绍如何利用Java反射机制来访问和操作父类的所有属性和方法。 #### 二、Java反射基础 在深入讨论如何操作父类之前,我们先简要回顾一下Java反射的基本概念: - **Class**: 每个类都有一个与之对应的`...
在本篇文章中,我们将深入探讨如何利用C#中的反射机制来访问类中的私有变量或方法。通过一个具体的示例代码,我们将详细了解反射的基本概念、使用场景以及如何具体操作。 ### 一、C#反射机制简介 #### 1.1 什么是...
### Java反射技术实战 #### 一、Java反射技术概述 Java反射技术是Java语言的一个重要特性,它允许程序在运行时动态地获取类的信息,并直接操作这些信息。这种能力为开发人员提供了极大的灵活性,同时也带来了一定...
首先,我们需要理解什么是Java反射机制。在Java中,当程序运行时,如果一个类的名字已知,但它的具体信息(如属性、方法)未知,反射机制就能帮助我们获取这些信息。这主要通过`java.lang.Class`类和`java.lang....
在Android开发中,反射机制是一种强大的工具,它允许我们在运行时动态地获取类的信息或调用私有方法、访问私有变量,甚至修改类的行为。本文将深入探讨反射的概念、工作原理以及如何在Android中使用反射,通过一个...
7. **读取或设置私有属性的值**。 8. **获取私有方法的信息**。 9. **调用私有方法**。 示例代码如下: ```csharp MyClass mc = new MyClass(); Type t = typeof(MyClass); BindingFlags bf = BindingFlags....
假设我们有两个 `Person` 类的实例 `person1` 和 `person2`,我们想要将 `person1` 的属性值复制到 `person2`: ```java public static void copyBeanProperties(Object source, Object target) throws ...
Java反射是Java编程语言中的一个...总之,Java反射提供了一种强大的机制,让我们能够在运行时动态地操控类和对象,实现代码的灵活性和扩展性。理解并熟练掌握反射技术,能够帮助开发者更好地应对各种复杂的需求和场景。
如果只想获取特定访问类型的属性,可以传递一个标志参数,如`ReflectionProperty::IS_PRIVATE`来只获取私有属性。 3. **方法 Method Names**: 类的方法信息可以通过`ReflectionClass::getMethods()`获取,同样可以...
- **浅度克隆**:仅复制对象的基本类型属性值,对于对象引用类型属性,只复制引用本身而非引用指向的对象。 - **深度克隆**:不仅复制对象本身,还递归复制对象内的所有引用对象。 **1.5 示例代码** ```java ...
// 获取私有属性值 $property = $reflection->getProperty('value'); $property->setAccessible(true); echo $property->getValue($instance); // 输出 42 // 调用私有方法 $method = $reflection->getMethod('...
以下是对标题和描述中提及的Java反射知识点的详细解释: 1. **获取对象的属性**: 在Java中,通过反射可以获取对象的属性。在示例代码中,`getProperty`方法展示了如何做到这一点。首先,通过`owner.getClass()`...
总之,Java反射与动态加载是Java平台的两大特性,它们提供了强大的元数据访问能力和运行时的灵活性。合理使用可以极大地提升程序的可扩展性和维护性,但也需要注意性能和安全问题。在实际开发中,应根据需求权衡利弊...
在C#编程中,反射(Reflection)是一种强大的机制,它允许程序在运行时检查自身的行为和结构。通过反射,开发者可以动态地创建对象、访问成员(如方法、属性和字段)以及获取类型信息,即使在编译时不知道具体的类型...
通过反射访问私有属性的方法与访问字段类似,但需要使用`GetProperty`方法: ```csharp // 获取私有属性 PropertyInfo pi_private = t.GetProperty("PrivateProperty", BindingFlags.Instance | BindingFlags....
2. **访问属性**:通过`Class.getDeclaredFields()`获取所有字段,然后通过`Field.get()`或`Field.set()`读取或设置属性值。 3. **调用方法**:通过`Class.getMethod()`或`Class.getDeclaredMethod()`获取指定方法,...