private的意义
-
https://mp.weixin.qq.com/s/mOctaV1QX4ju1lUS8z7w-Q
-
面试官:private修饰的方法可以通过反射访问,那么private的意义是什么?
不让外界访问到某些属性和方法,通常将其设置为private,
用正常的方式(对象名.属性名,对象名.方法名)将无法访问此属性与方法,
但有没有其他方法可以访问呢?答案是有的,这就是java反射带来的便利。
- 利用反射访问类的私有属性及方法如下:
java bean
public class Reflect {
//私有属性
private String name;
private int age;
//私有构造
private Reflect(int age) {
this.age = age;
}
//共有构造
public Reflect(String name) {
this.name = name;
}
//私有方法
private void speak(String name) {
System.out.println("My name is" + name);
}
}
三个反射包中的类
首先,我们要了解三个反射包中的类:
-
Constructor:代表类的单个构造方法,
- 通过Constructor我们可执行一个类的某个构造方法(有参或者无参)来创建对象时。
-
Method:代表类中的单个方法,
- 可以用于执行类的某个普通方法,有参或无参,并可以接收返回值。
-
Field:代表类中的单个属性,
- 用于set或get属性
-
AccessibleObject:以上三个类的父类,提供了构造方法,普通方法,和属性的访问控制的能力。
使用Class类中的方法可以获得该类中的所有Constructor对象,Method对象,和Field对象。
但是仍然无法访问
- 私有化的构造方法,普通方法,和私有属性,
- 此时我们可以使用他们 继承父类
- (AccessibleObject)中的setAccessible()方法,来设置或取消访问检查,以达到访问私有对象的目的。
进行测试
- 结果失败了。不能访问 private方法(改为public有这个方法),但是invoke不执行具体方法
//定义反射对象,name为a
Reflect reflect = new Reflect("a");
//通过反射 拿到所有的方法
Method[] methods = Reflect.class.getMethods();
//拿到 所有的字段
Field[] fields = Reflect.class.getDeclaredFields();
//遍历所有的字段,访问权限设置为true
for (int i = 0; i < fields.length; i++) {
fields[i].setAccessible(true);
//输出 每个字段的名字
System.out.println(fields[i].getName());
}
//遍历所有的方法
for (int j = 0; j < methods.length; j++) {
//设置 可以访问
methods[j].setAccessible(true);
//打印方法的名字
System.out.println(methods[j].getName());
try {
//执行这个方法,name为a
methods[j].invoke(reflect);
} catch (Exception e) {
//e.printStackTrace();
}
//System.out.println("再次打印方法名:" + methods[j].getName());
}
name
age
wait
wait
wait
equals
toString
hashCode
getClass
notify
notifyAll
这样,我们就获得了私有属性的值
当然,凡事有利就有弊,然后我们再来说一下java反射的优缺点;
优点
- 能够运行时动态获取类的实例,大大提高了系统的灵活性和扩展性;
- 与java动态编译相结合,可以实现无比强大的功能。
缺点
- 使用反射的性能较低;
- 使用反射来说相对不安全;
- 破坏了类的封装性,可以通过反射来获取这个类的属性,和私有方法。
意义
private修饰的方法可以通过反射访问,那么private的意义是什么?
- Java的private修饰符不是为了绝对安全设计的,
- 而是对用户常规使用Java的一种约束。
- 就好比饭店厨房门口挂着“闲人免进”的牌子,但是你还是能够通过其他方法进去。
- 从外部对对象进行常规调用时,能够看到清晰的类结构