上周在修复bug时,发现Java类中某方法是private,且类中没有用到,第一感觉是方法多余。其实通过分析,发现原来Native Code会通过JNI调到此方法。这也给自己启发,平时做Code refine时,如果方法没有被直接调用,一定要注意是否会有反射会调用到等,否则移除后,编译通过,但会有问题~
虽然是小语法点,但很感兴趣,而且易被忽略。这两天查了一些资料,分享下
一. Native Code调用Java类中private方法
例如JNI开发中,C可以反射调到Java的方法,例如通过GetMethodID等API,原因解释:
![分享调用Java private方法 分享调用Java private方法](https://image.shishitao.com:8440/aHR0cHM6Ly9iYnNtYXguaWthZmFuLmNvbS9zdGF0aWMvTDNCeWIzaDVMMmgwZEhCekwybHRZV2RsY3pJd01UY3VZMjVpYkc5bmN5NWpiMjB2WW14dlp5ODFOVFk0TVRFdk1qQXhOekE0THpVMU5qZ3hNUzB5TURFM01EZ3dNVEU0TXpZek9EY3dPQzB4TVRJd056QXlNREV6TG5CdVp3PT0uanBn.jpg?w=700)
所以在判断private方法在哪里被调用时,记得多在代码里搜索下,避免出现此类情况,误认为方法没有被使用~
参考:
二. Java中通过反射也可以调用其他类的private方法
举例:
![分享调用Java private方法 分享调用Java private方法](https://image.shishitao.com:8440/aHR0cHM6Ly9iYnNtYXguaWthZmFuLmNvbS9zdGF0aWMvTDNCeWIzaDVMMmgwZEhCekwybHRZV2RsY3pJd01UY3VZMjVpYkc5bmN5NWpiMjB2WW14dlp5ODFOVFk0TVRFdk1qQXhOekE0THpVMU5qZ3hNUzB5TURFM01EZ3dNVEU0TXpnMU5EY3lOQzB4TURjeU9EQTROVGt6TG5CdVp3PT0uanBn.jpg?w=700)
其中a是Test类中的private方法,通过getDeclaredMethod可以获得目标Class中的方法(不包含父类)。能否执行private方法,取决于setAccessible API,此接口会在基类AccessObject中设置成员变量overide为true,注释的解释很有用,如下:
![分享调用Java private方法 分享调用Java private方法](https://image.shishitao.com:8440/aHR0cHM6Ly9iYnNtYXguaWthZmFuLmNvbS9zdGF0aWMvTDNCeWIzaDVMMmgwZEhCekwybHRZV2RsY3pJd01UY3VZMjVpYkc5bmN5NWpiMjB2WW14dlp5ODFOVFk0TVRFdk1qQXhOekE0THpVMU5qZ3hNUzB5TURFM01EZ3dNVEU0TXpreU9UQXlNUzB4TnpFMU56QTFNemsyTG5CdVp3PT0uanBn.jpg?w=700)
如overide为true,会取消Language-Level的安全检查,如访问修饰符private的限制等。另外出于安全考虑,非此package无法访问overide变量。
最后分享个小的tip,Android中反射相关代码来自ojluni,如果对这个名字感到不解的话,它的意思是: OpenJDK;java.lang;java.util;java.net ;java.io 的缩写,就是OpenJDK核心库的意思,与Google采用OpenJDK代替原有Java API的变动有关~Thanks
- Kevin Song
2017.8.1