Java语言中,在一个类中,为了不让外界访问到有的属性和方法,通常将其设置为private,用正常的方式(对象名.属性名,对象名.方法名)将无法访问此属性与方法,但有没有其他方法可以访问呢?答案是有的,这就是java反射带来的便利。利用反射访问类的私有属性及方法如下:

1.准备一个java类,包含私有属性及方法:

  1. //Exam.java
  2. public class Exam{
  3. private String field1="私有属性";
  4. public String field2="公有属性";
  5. public void fun1(){
  6. System.out.println("fun1:这是一个public访问权限方法");
  7. }
  8. private void fun2(){
  9. System.out.println("fun2:这是一个private访问权限方法");
  10. }
  11. private void fun3(String arg){
  12. System.out.println("fun3:这是一个private访问权限且带参数的方法,参数为:"+arg);
  13. }
  14. }

将其编译成class,然后删除java源文件。注意:删除java源文件并非必须,但是在实际情况中,我会使用的往往不是java源文件,而是jar包,而jar包中的文件都是class,所以为了贴近实际的情况,将Exam.java编译成Exam.class文件后,删除Exam.java文件,只保留Exam.class文件。

2.获取类中属性及方法的信息

第一步做好后,接下来进行第二步:获取类中属性及方法的信息。对于一个class文件,无法查看源码,如何获取类中属性及方法的信息呢?很自然地,我们会想到利用java反射功能,得到类的属性、方法:

  1. //Test01.java
  2. import java.lang.reflect.Field;
  3. import java.lang.reflect.InvocationTargetException;
  4. import java.lang.reflect.Method;
  5. public class Test01 {
  6. public static void main(String args[]){
  7. Exam e=new Exam();  //初始化Exam实例
  8. System.out.println("获取类中所有的属性:");
  9. Field field[] = e.getClass().getFields();
  10. for(Field f : field){
  11. System.out.println(f);
  12. }
  13. System.out.println("获取类中所有的方法:");
  14. Method[] method = e.getClass().getMethods();
  15. for(Method m : method){
  16. System.out.println(m);
  17. }
  18. }
  19. }

运行结果如下:

从运行结果可以看到,获取的属性中只有field2,而获取的方法中似乎多了好多类中没有的,但最主要的是,无论是属性还是方法,都是public声明的,没有得到private声明的属性或方法。由此可以推断:

1.声明为private的属性或方法无法通过这种途径来获取。

2.结果中多出来的方法为从Object类中继承来的方法。

那么,是不是就没有办法了呢?不是的!我们可以使用jdk中的javap命令来突破这个问题:

java -private Exam                                                                ; -private 标志表示所有的成员都应该显示,甚至包括私有成员

运行结果如下:

此时,可以看到类中所有的属性、方法都有了。

3.调用Method及Field类中的相关方法获取private声明的属性及方法

也许你还在为第二步中如此辛苦地得到类中的信息相当不解,其实这是在为这一步作铺垫。请看以下代码:Testo2.java

  1. import java.lang.reflect.Field;
  2. import java.lang.reflect.InvocationTargetException;
  3. import java.lang.reflect.Method;
  4. public class Test02 {
  5. public static void main(String args[]){
  6. Exam e=new Exam();
  7. try {
  8. field1 = e.getClass().getDeclaredField("field1");
  9. field2 = e.getClass().getDeclaredField("field2");
  10. field1.setAccessible(true);
  11. System.out.println("field1: "+field1.get(e));
  12. field1.set(e,"重新设置一个field1值");
  13. System.out.println("field1: "+field1.get(e));
  14. System.out.println("field2: "+field2.get(e));
  15. field2.set(e,"重新设置一个field2值");
  16. System.out.println("field2: "+field2.get(e));
  17. } catch (NoSuchFieldException e1) {
  18. e1.printStackTrace();
  19. }catch (IllegalArgumentException e1) {
  20. e1.printStackTrace();
  21. } catch (IllegalAccessException e1) {
  22. e1.printStackTrace();
  23. }
  24. try {
  25. Method method1 = e.getClass().getDeclaredMethod("fun1");
  26. method1.invoke(e);
  27. Method method2 = e.getClass().getDeclaredMethod("fun2");
  28. method2.setAccessible(true);
  29. method2.invoke(e);
  30. Method method3 = e.getClass().getDeclaredMethod("fun3",String.class);
  31. method3.setAccessible(true);
  32. method3.invoke(e,"fun3的参数");
  33. } catch (NoSuchMethodException e1) {
  34. // TODO Auto-generated catch block
  35. e1.printStackTrace();
  36. } catch (SecurityException e1) {
  37. // TODO Auto-generated catch block
  38. e1.printStackTrace();
  39. }catch (IllegalAccessException e1) {
  40. // TODO Auto-generated catch block
  41. e1.printStackTrace();
  42. } catch (IllegalArgumentException e1) {
  43. // TODO Auto-generated catch block
  44. e1.printStackTrace();
  45. } catch (InvocationTargetException e1) {
  46. // TODO Auto-generated catch block
  47. e1.printStackTrace();
  48. }
  49. }
  50. }

注意以下语句:

  1. field1 = e.getClass().getDeclaredField("field1");

getDeclaredField(String fieldName)中,参数fieldName为属性名,

  1. Method method3 = e.getClass().getDeclaredMethod("fun3",String.class);

getDeclaredMethod(String methodName,Class parameterType)中,第一个参数为方法名,第二个参数为方法参数类型,当然在此方法中第二个参数为可娈参数。

Test02.java运行结果如下:

由运行结果可知,利用反射不但可以访问类的私有属性、方法,还可以重新设置私有属性的值,调用私有方法。

java利用反射访问类的私有(private)属性及方法的更多相关文章

  1. java利用反射获取类的属性及类型

    java利用反射获取类的属性及类型. import java.lang.reflect.Field; import java.math.BigDecimal; import java.util.Map ...

  2. Java利用反射取得类的所有信息

    Java中可以利用反射获取类的名称.构造函数.属性.方法.也就是说可以通过反射可以取得类的所有信息(不管该成员是否封装为private). 如有下面的Dept类定义: package org.lyk. ...

  3. java利用反射调用类的某个方法

    java利用反射机制 可以动态调用某个类的某个方法,在 扩展系统功能或提供对外接口时经常用的到. 代码如下: 打印类Print.java package com.test.reflct; /** * ...

  4. Java第二天,类的概念,属性和方法的使用

    上文中我们已近说到过了,Java是一种面向对象的编程语言,对象是用类来创建的,就比如世界上有无数个父亲,但是他们都有一个共同的属性--男人.也就是说某个父亲这个对象属于男人这个类.类是Java必不可少 ...

  5. 【转】Java利用反射机制访问私有化构造器

    Java利用反射机制访问私有化构造器 博客分类: java   我们都知道,当一个类的构造方法被设为私有的时候(private),在其他类中是无法用new来实例化一个对象的. 但是有一种方法可以把带有 ...

  6. java中用反射访问私有方法和私有成员[转]

    转自: http://zhouyangchenrui.iteye.com/blog/470521 java的反射可以绕过访问权限,访问到类的私有方法和成员.可能这点会引起安全性的讨论.反射的使用帮助解 ...

  7. 利用Java反射根据类的名称获取属性信息和父类的属性信息

    代码: import java.lang.reflect.Field; import java.util.ArrayList; import java.util.Arrays; import java ...

  8. Delphi 跨单元进入(访问)类的私有成员,protected ,private部分

    http://blog.sina.com.cn/s/blog_5f8861b60102v1nl.html Delphi 跨单元进入(访问)类的私有成员,protected ,private部分 (20 ...

  9. Java -- JDBC_利用反射及 JDBC 元数据编写通用的查询方法

    先利用 SQL 进行查询,得到结果集: 利用反射创建实体类的对象:创建对象: 获取结果集的列的别名: 再获取结果集的每一列的值, 结合 3 得到一个 Map,键:列的别名,值:列的值: 再利用反射为 ...

随机推荐

  1. Codeforces Round #324 (Div. 2) A B C D E

    A,水题不多说. #include<bits/stdc++.h> using namespace std; //#define LOCAL int main() { #ifdef LOCA ...

  2. js中随机数获取

    // 结果为0-1间的一个随机数(包括0,不包括1) var randomNum1 = Math.random(); //console.log(randomNum1); // 函数结果为入参的整数部 ...

  3. CUDA常见问题与解答

    源 1.在SDK自带的例子程序中,发现SRC文件珜下有.cpp文件和.cu文件.这两种文件的关系和各自的作用是什么呀? 答:SDK自带例子中的.cpp文件主要是一些CPU端处理,或者是使用CPU计算对 ...

  4. 微信小程序的开发——01小程序的执行流程是怎样的?

    作者:叶小钗 转载至:https://www.cnblogs.com/yexiaochai/p/9346043.html 我们这边最近一直在做基础服务,这一切都是为了完善技术体系,这里对于前端来说便是 ...

  5. OI,我的决心

    虽然从初一就开始NOIP,但沉溺于游戏编程等各种乱七八糟的技术,一直没对算法有过透彻的研究. ——————————简单的来说就是水过了—————————— 我生于一个弱省,就读于一所弱校(我们全区的都 ...

  6. build path导入的jar失效导致找不到类

    今天碰到一个很奇葩的问题,搞起我以后都不敢 build path到jar了 所以我就全部放到lib目录下了,因为之前使用build path导入的jar失效了,一直找不类,具体原因我也不清楚,我之前的 ...

  7. vue2.0:子组件调用父组件

    main.js文件添加如下: new Vue({ router, render: h => h(App), data: { eventHub: new Vue() }}).$mount('#ap ...

  8. mem_init()

    原本由bootmem管理的内存在mem_init函数中交由伙伴系统管理. 1.free_unused_memmap_node 相邻的membank间可能存在空洞,但在bootmem阶段这些空洞页也分配 ...

  9. Tame Me【驯服我】

    Tame Me “Good morning,” said the fox. 早上好,狐狸说 “Good morming,” the little prince responded politely,a ...

  10. Codeforces Round #456 (Div. 2) B. New Year's Eve

    传送门:http://codeforces.com/contest/912/problem/B B. New Year's Eve time limit per test1 second memory ...