java 反射父类私有属性_如何证明Java子类实际上是拥有父类的私有属性

前言

最近在看JavaGuide进行基础知识的回顾,注意到一个有意思的观点,原文如下

关于继承如下 3 点请记住:

子类拥有父类对象所有的属性和方法(包括私有属性和私有方法),但是父类中的私有属性和方法子类是无法访问,只是拥有。

于是我就开始尝试验证这一观点

对象初始化

首先从对象初始化开始思考,一般继承某个父类的子类对象初始化时是按照以下顺序(对这块感兴趣的可以给我留言 )

父类的静态变量和常量以及父类的静态代码块

子类的静态变量和常量以及子类的静态代码块

父类的变量赋默认值和父类代码块

父类的构造方法

子类的变量赋默认值和父类代码块

子类的构造方法

既然按照对象初始化顺序来说,子类的构造方法被调用的时候必然有一个父类对象被构造,那么必然有办法在子类构造时对父类的变量进行赋值。

这个时候我突然想到了Java如果不指定构造方法,会默认给每个类提供一个无参构造方法,那如果我的父类只有有参构造方法又会如何呢?

使用有参构造方法

首先我准备好了父类,这个父类里面只有一个parentName的属性,同时制定一个有参构造方法,代码如下:

public class Parent {

private String parentName;

public Parent(String parentName) {

this.parentName = parentName;

}

}

然后我准备了子类继承父类,这个时候有意思的事情发生了,子类必须提供一个带参的构造方法来构造父类对象,同时要在这个带参构造方法里面通过super关键字调用父类的构造方法,否则编译会失败,最终代码如下:

public class Child extends Parent{

public Child(String parentName) {

super(parentName);

}

}

其实从这里就可以知道子类是拥有父类的私有属性的了,只是因为访问控制限制无法直接访问这个属性。

但是除了这个方法以外,我还有什么办法去佐证这个观点呢?当然有啦,而且还不止一种呢,我知道的就有两个办法

直接查看字节码

使用反射获取

使用反射获取

由于篇幅问题,这里就展示怎么使用反射获取父类的属性的示例代码,如果对通过字节码验证这个观点感兴趣可以给我留言:

Child child = new Child("David");

try {

// 获取父类Class对象

Class parentClass = child.getClass().getSuperclass();

// 获取父类的属性

Field field = parentClass.getDeclaredField("parentName");

// 解除访问限制

field.setAccessible(true);

// 输出

System.out.println(field.get(child));

} catch (NoSuchFieldException e) {

// 这里只是演示用,实际开发时需要将错误信息打印到日志文件

e.printStackTrace();

} catch (IllegalAccessException e) {

e.printStackTrace();

}

输出结果

David

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要通过Java反射获取父类子类的所有属性,我们可以使用Class类的getFields()和getDeclaredFields()方法。 - getFields()方法可以获取类中所有公共的(即用public修饰的)属性,包括父类中的公共属性。返回的是一个数组,其中包含了所有公共属性的Field对象。 - getDeclaredFields()方法可以获取类中所有声明的属性,包括私有的、受保护的和默认访问权限的属性,但不包括父类中的属性。返回的也是一个数组,其中包含了所有声明的属性的Field对象。 我们可以先获取子类的Class对象,然后利用getFields()和getDeclaredFields()方法获取子类自己声明的属性和从父类继承的公共属性。 再获取父类的Class对象,利用getFields()方法获取父类的公共属性。 下面是一个示例代码: ```java import java.lang.reflect.Field; public class ReflectionExample { public static void main(String[] args) { Child child = new Child(); // 获取子类Class对象 Class<?> childClass = child.getClass(); // 获取子类自己声明的属性和从父类继承的公共属性 Field[] childFields = childClass.getDeclaredFields(); for (Field field : childFields) { System.out.println(field.getName()); } // 获取父类Class对象 Class<?> parentClass = childClass.getSuperclass(); // 获取父类的公共属性 Field[] parentFields = parentClass.getFields(); for (Field field : parentFields) { System.out.println(field.getName()); } } } class Parent { public int parentField; } class Child extends Parent { private String childField; } ``` 在上面的示例中,我们创建了一个Parent类和一个Child类。Child类继承自Parent类,并在自己中声明了一个private的childField属性。 运行示例代码,结果会打印出子类父类的所有属性名: ``` childField parentField ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值