参考答案
可以。
Java语言中,子类不能访问父类的私有成员,包括成员变量和方法。但是,通过Java的反射机制,则可以访问父类的成员。
下面用两个实例,来演示如何用反射方式获取父类的属性的值。
实例一
父类:
package date0415.pm.反射; public class TestParent{ private String str1="www";//初始值。 private String str2="hao123";//初始值。 private int int1; public TestParent() { super(); } public TestParent(String str1, String str2,int int1) { super(); this.str1 = str1; this.str2 = str2; this.int1 = int1; } /** * 打印方法,负责打印该类变量的值 */ public void print(){ System.out.println(str1); System.out.println(str2); System.out.println(int1); } }
子类:
package date0415.pm.反射; import java.lang.reflect.Field; public class TestChild extends TestParent{ public TestChild() { super(); } public TestChild(String str1, String str2, int mVerticalSpacing) { super(str1, str2, mVerticalSpacing); } /** * 获取指定字段的值(是一个对象) */ public Object getSpecificedFieldObject(String fieldName) { Class<?> clazz = this.getClass().getSuperclass(); Object object = null; try { Field field = clazz.getDeclaredField(fieldName); field.setAccessible(true); object = field.get(this); } catch (NoSuchFieldException e) { e.printStackTrace(); } catch (SecurityException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } return object; } }
主方法类:
package date0415.pm.反射; public class TestMain{ public static void main(String[] args){ // TestParent t=new TestParent(); // t.print(); // // Field[] f = t.getClass().getDeclaredFields(); // // for(int i=0;i<f.length;i++){ // f[i].setAccessible(true); // if (f[i].getName() == "str2") { // f[i].set(t, "japan"); // }; // } // t.print(); test1(); System.out.println("\n\n---------分割线--------\n\n"); test2(); } /** * 测试方法1 */ private static void test1(){ TestChild child = new TestChild(); String v1 = (String) child.getSpecificedFieldObject("str1"); String v2 = (String) child.getSpecificedFieldObject("str2"); int v3 = (int) child.getSpecificedFieldObject("int1"); System.out.println("反射取值:"); System.out.println("str1 = "+v1); System.out.println("str2 = "+v2); System.out.println("str3 = "+v3); System.out.println("\n调用继承自父类的方法打印:"); child.print();//调用父类方法打印值 } /** * 测试方法2 */ private static void test2(){ TestChild child = new TestChild("你好", "世界", 10086); String v1 = (String) child.getSpecificedFieldObject("str1"); String v2 = (String) child.getSpecificedFieldObject("str2"); int v3 = (int) child.getSpecificedFieldObject("int1"); System.out.println("反射取值:"); System.out.println("str1 = "+v1); System.out.println("str2 = "+v2); System.out.println("str3 = "+v3); System.out.println("\n调用继承自父类的方法打印:"); child.print();//调用父类方法打印值 } }
结果:
反射取值: str1 = www str2 = hao123 str3 = 0 调用父类方法打印: www hao123 0 ---------分割线-------- 反射取值: str1 = 你好 str2 = 世界 str3 = 10086 调用父类方法打印: 你好 世界 10086
实例二
父类:
/** * 父类 */ public class Parent { public String publicField = "1"; String defaultField = "2"; protected String protectedField = "3"; private String privateField = "4" ; public void publicMethod() { System.out.println("publicMethod..."); } void defaultMethod() { System.out.println("defaultMethod..."); } protected void protectedMethod() { System.out.println("protectedMethod..."); } private void privateMethod() { System.out.println("privateMethod..."); } }
子类:
/** * 子类 */ public class Son extends Parent{ }
反射:
import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; /** * 方法类 */ public class ReflectionUtils { /** * 循环向上转型, 获取对象的 DeclaredMethod * @param object : 子类对象 * @param methodName : 父类中的方法名 * @param parameterTypes : 父类中的方法参数类型 * @return 父类中的方法对象 */ public static Method getDeclaredMethod(Object object, String methodName, Class<?> ... parameterTypes){ Method method = null ; for(Class<?> clazz = object.getClass() ; clazz != Object.class ; clazz = clazz.getSuperclass()) { try { method = clazz.getDeclaredMethod(methodName, parameterTypes) ; return method ; } catch (Exception e) { //这里甚么都不要做!并且这里的异常必须这样写,不能抛出去。 //如果这里的异常打印或者往外抛,则就不会执行clazz = clazz.getSuperclass(),最后就不会进入到父类中了 } } return null; } /** * 直接调用对象方法, 而忽略修饰符(private, protected, default) * @param object : 子类对象 * @param methodName : 父类中的方法名 * @param parameterTypes : 父类中的方法参数类型 * @param parameters : 父类中的方法参数 * @return 父类中方法的执行结果 */ public static Object invokeMethod(Object object, String methodName, Class<?> [] parameterTypes, Object [] parameters) { //根据 对象、方法名和对应的方法参数 通过反射 调用上面的方法获取 Method对象 Method method = getDeclaredMethod(object, methodName, parameterTypes); //抑制Java对方法进行检查,主要是针对私有方法而言 method.setAccessible(true) ; try { if(null != method) { //调用object 的 method 所代表的方法,其方法的参数是 parameters return method.invoke(object, parameters) ; } } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } return null; } /** * 循环向上转型, 获取对象的 DeclaredField * @param object : 子类对象 * @param fieldName : 父类中的属性名 * @return 父类中的属性对象 */ public static Field getDeclaredField(Object object, String fieldName){ Field field = null ; Class<?> clazz = object.getClass() ; for(; clazz != Object.class ; clazz = clazz.getSuperclass()) { try { field = clazz.getDeclaredField(fieldName) ; return field ; } catch (Exception e) { //这里甚么都不要做!并且这里的异常必须这样写,不能抛出去。 //如果这里的异常打印或者往外抛,则就不会执行clazz = clazz.getSuperclass(),最后就不会进入到父类中了 } } return null; } /** * 直接设置对象属性值, 忽略 private/protected修饰符, 也不经过 setter * @param object : 子类对象 * @param fieldName : 父类中的属性名 * @param value : 将要设置的值 */ public static void setFieldValue(Object object, String fieldName, Object value) { //根据 对象和属性名通过反射 调用上面的方法获取 Field对象 Field field = getDeclaredField(object, fieldName) ; //抑制Java对其的检查 field.setAccessible(true) ; try { //将 object 中 field 所代表的值 设置为 value field.set(object, value) ; } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } } /** * 直接读取对象的属性值, 忽略 private/protected修饰符, 也不经过 getter * @param object : 子类对象 * @param fieldName : 父类中的属性名 * @return : 父类中的属性值 */ public static Object getFieldValue(Object object, String fieldName){ //根据 对象和属性名通过反射 调用上面的方法获取 Field对象 Field field = getDeclaredField(object, fieldName) ; //抑制Java对其的检查 field.setAccessible(true); try { //获取 object 中 field 所代表的属性值 return field.get(object); } catch(Exception e) { e.printStackTrace(); } return null; } }
反射测试:
import static org.junit.Assert.*; import java.lang.reflect.Field; import java.lang.reflect.Method; import org.junit.Test; /** * 测试类,用JUnit4 进行测试 */ public class ReflectionUtilsTest { /** * 测试获取父类的各个方法对象 */ @Test public void testGetDeclaredMethod() { Object obj = new Son() ; //获取公共方法名 Method publicMethod = ReflectionUtils.getDeclaredMethod(obj, "publicMethod") ; System.out.println(publicMethod.getName()); //获取默认方法名 Method defaultMethod = ReflectionUtils.getDeclaredMethod(obj, "defaultMethod") ; System.out.println(defaultMethod.getName()); //获取被保护方法名 Method protectedMethod = ReflectionUtils.getDeclaredMethod(obj, "protectedMethod") ; System.out.println(protectedMethod.getName()); //获取私有方法名 Method privateMethod = ReflectionUtils.getDeclaredMethod(obj, "privateMethod") ; System.out.println(privateMethod.getName()); } /** * 测试调用父类的方法 * @throws Exception */ @Test public void testInvokeMethod() throws Exception { Object obj = new Son() ; //调用父类的公共方法 ReflectionUtils.invokeMethod(obj, "publicMethod", null , null) ; //调用父类的默认方法 ReflectionUtils.invokeMethod(obj, "defaultMethod", null , null) ; //调用父类的被保护方法 ReflectionUtils.invokeMethod(obj, "protectedMethod", null , null) ; //调用父类的私有方法 ReflectionUtils.invokeMethod(obj, "privateMethod", null , null) ; } /** * 测试获取父类的各个属性名 */ @Test public void testGetDeclaredField() { Object obj = new Son() ; //获取公共属性名 Field publicField = ReflectionUtils.getDeclaredField(obj, "publicField") ; System.out.println(publicField.getName()); //获取公共属性名 Field defaultField = ReflectionUtils.getDeclaredField(obj, "defaultField") ; System.out.println(defaultField.getName()); //获取公共属性名 Field protectedField = ReflectionUtils.getDeclaredField(obj, "protectedField") ; System.out.println(protectedField.getName()); //获取公共属性名 Field privateField = ReflectionUtils.getDeclaredField(obj, "privateField") ; System.out.println(privateField.getName()); } @Test public void testSetFieldValue() { Object obj = new Son() ; System.out.println("原来的各个属性的值: "); System.out.println("publicField = " + ReflectionUtils.getFieldValue(obj, "publicField")); System.out.println("defaultField = " + ReflectionUtils.getFieldValue(obj, "defaultField")); System.out.println("protectedField = " + ReflectionUtils.getFieldValue(obj, "protectedField")); System.out.println("privateField = " + ReflectionUtils.getFieldValue(obj, "privateField")); ReflectionUtils.setFieldValue(obj, "publicField", "a") ; ReflectionUtils.setFieldValue(obj, "defaultField", "b") ; ReflectionUtils.setFieldValue(obj, "protectedField", "c") ; ReflectionUtils.setFieldValue(obj, "privateField", "d") ; System.out.println("***********************************************************"); System.out.println("将属性值改变后的各个属性值: "); System.out.println("publicField = " + ReflectionUtils.getFieldValue(obj, "publicField")); System.out.println("defaultField = " + ReflectionUtils.getFieldValue(obj, "defaultField")); System.out.println("protectedField = " + ReflectionUtils.getFieldValue(obj, "protectedField")); System.out.println("privateField = " + ReflectionUtils.getFieldValue(obj, "privateField")); } @Test public void testGetFieldValue() { Object obj = new Son() ; System.out.println("publicField = " + ReflectionUtils.getFieldValue(obj, "publicField")); System.out.println("defaultField = " + ReflectionUtils.getFieldValue(obj, "defaultField")); System.out.println("protectedField = " + ReflectionUtils.getFieldValue(obj, "protectedField")); System.out.println("privateField = " + ReflectionUtils.getFieldValue(obj, "privateField")); } }
运行所有单元测试及结果:
publicField = 1 defaultField = 2 protectedField = 3 privateField = 4 原来的各个属性的值: publicField = 1 defaultField = 2 protectedField = 3 privateField = 4 *********************************************************** 将属性值改变后的各个属性值: publicField = a defaultField = b protectedField = c privateField = d publicMethod... defaultMethod... protectedMethod... privateMethod... publicMethod defaultMethod protectedMethod privateMethod publicField defaultField protectedField privateField
建议尝试下~
以上,是Java面试题【 Java反射可以访问父类的成员吗】的参考答案。
输出,是最好的学习方法。
立即行动,在评论区记录下你的问题、笔记或补充~
—end—