1 public class Demo12{ 2 3 /* 4 对任意一个Object实例,只要我们获取了它的Class,就可以获取它的一切信息 5 Class类提供了以下几个方法来获取字段 6 7 Field getField(name):根据字段名获取某个public的field(包括父类) 8 Field getDeclaredField(name):根据字段名获取当前类的某个field(不包括父类) 9 Field getFields():获取全部public的field(包括父类) 10 Field getDeclaredField():获取当前类的全部field(不包括父类) 11 */ 12 13 //reflection反射 14 public static void main(String[] args) throws Exception{ //记住要抛出异常 15 Class stuClass = Student.class; 16 //获取public字段"score" 17 System.out.println(stuClass.getField("score"));//public int Student.score 18 //获取继承的public字段"name" 19 System.out.println(stuClass.getField("name"));//public java.lang.String Person.name 20 //获取private字段"grade" 21 System.out.println(stuClass.getDeclaredField("grade")); 22 } 23 } 24 25 class Student extends Person{ 26 public int score; 27 //班级字段私有 28 private int grade; 29 } 30 31 class Person{ 32 public String name; 33 }
一个Field对象包含了一个字段的所有信息
getName():返回字段名称,例如"name"
getType():返回字段类型,也是一个Class实例,例如,String.class
getModifiers():返回字段的修饰符,它是一个int,不同的bit表示不同的含义
1 public static void main(String[] args) throws Exception{ //记住要抛出异常 2 Class stuClass = Student.class; 3 //获取public字段"score" 4 Field f = stuClass.getField("score"); 5 //获取字段名称 6 System.out.println(f.getName());//score 7 //获取字段类型 8 System.out.println(f.getType()); 9 //获取字段的修饰符所代表的bit数字 10 int m = f.getModifiers(); 11 System.out.println(m); 12 //判断字段是哪种修饰符 13 System.out.println(Modifier.isFinal(m)); 14 System.out.println(Modifier.isPublic(m)); 15 System.out.println(Modifier.isProtected(m)); 16 System.out.println(Modifier.isPrivate(m)); 17 System.out.println(Modifier.isStatic(m)); 18 }
获取字段值
利用反射拿到字段的一个Field
实例只是第一步,我们还可以拿到一个实例对应的该字段的值。
例如,对于一个Person
实例,我们可以先拿到name
字段对应的Field
,再获取这个实例的name
字段的值:
1 import java.lang.reflect.*; 2 public class Demo12{ 3 //reflection反射 4 public static void main(String[] args) throws Exception{ //记住要抛出异常 5 Object p = new Person("先把SE学完!"); 6 Class cls = p.getClass(); 7 Field f = cls.getDeclaredField("name"); 8 //因为name被定义为一个private字段,正常情况下,Main类无法访问Person类的private字段,会得到一个IllegalAccessException 9 //调用Field.setAccessible(true)的意思是,别管这个字段是不是public,一律允许访问 10 f.setAccessible(true); 11 12 //用Field.get(Object)获取指定实例的指定字段的值 13 Object value = f.get(p); 14 System.out.println(value);//先把SE学完! 15 } 16 } 17 18 class Person { 19 private String name; 20 21 public Person(String name) { 22 this.name = name; 23 } 24 }
有童鞋会问:如果使用反射可以获取private
字段的值,那么类的封装还有什么意义?
答案是正常情况下,我们总是通过p.name
来访问Person
的name
字段,编译器会根据public
、protected
和private
决定是否允许访问字段,这样就达到了数据封装的目的。
而反射是一种非常规的用法,使用反射,首先代码非常繁琐,其次,它更多地是给工具或者底层框架来使用,目的是在不知道目标实例任何信息的情况下,获取特定字段的值。
此外,setAccessible(true)
可能会失败。如果JVM运行期存在SecurityManager
,那么它会根据规则进行检查,有可能阻止setAccessible(true)
。例如,某个SecurityManager
可能不允许对java
和javax
开头的package
的类调用setAccessible(true)
,这样可以保证JVM核心库的安全。
设置字段值
通过Field实例既然可以获取到指定实例的字段值,自然也可以设置字段的值。
设置字段值是通过Field.set(Object, Object)
实现的,其中第一个Object
参数是指定的实例,第二个Object
参数是待修改的值。示例代码如下:
1 import java.lang.reflect.*; 2 public class Demo12{ 3 //reflection反射 4 public static void main(String[] args) throws Exception{ //记住要抛出异常 5 //创建Person对象 6 Object p = new Person("正在学反射"); 7 //获取Person对象的Class对象 8 Class cls = p.getClass(); 9 //获取Class对象的字段对象 10 Field f = cls.getDeclaredField("name"); 11 //因为字段是private,所以必须设置访问权限为true,才可以进行访问 12 f.setAccessible(true); 13 //访问字段值 14 Object value = f.get(p); 15 System.out.println(value); 16 //修改字段值 17 f.set(p,"坚持学下去!"); 18 //再次访问字段值 19 value = f.get(p); 20 System.out.println(value); 21 } 22 } 23 24 class Person { 25 private String name; 26 27 public Person(String name) { 28 this.name = name; 29 } 30 }
小结
Java的反射API提供的Field
类封装了字段的所有信息:
通过Class
实例的方法可以获取Field
实例:getField()
,getFields()
,getDeclaredField()
,getDeclaredFields()
;
通过Field实例可以获取字段信息:getName()
,getType()
,getModifiers()
;
通过Field实例可以读取或设置某个对象的字段,如果存在访问限制,要首先调用setAccessible(true)
来访问非public
字段。
通过反射读写字段是一种非常规方法,它会破坏对象的封装。