我是靠谱客的博主 坚强凉面,最近开发中收集的这篇文章主要介绍对java Field中的set()方法以及Modifier的一些思考,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

只前转载了一篇文章《JAVA反射修改常量,以及其局限》,受益匪浅,在这里再次感谢原作者。

在看完文章后,我又试着写了一个测试程序,在期间也遇到了一些问题,写在这里,供大家参考。

程序入下:

import java.lang.reflect.Field;
class My{
private static final int value= 1;
private static String str="sds";
public
static int getValue()
{
return value;
}
public String getString()
{
return str;
}
}
public class FinalReflect{
public static void main(String[] args) throws Exception
{
Class c = Class.forName("reflect.My");
My M = (My)c.newInstance();
Field f = c.getDeclaredField("value");
Field s=c.getDeclaredField("str");
s.setAccessible(true);
f.setAccessible(true);
//
f.set(M,5);//对于static final 变量这里会出错!
s.set(null ,"ss");
System.out.println("str: "+s.get(M));
System.out.println("Field.get:"+f.get(M));
System.out.println(M.getString());
System.out.println(M.getValue());
}
}
一个问题是,当我将value设为static final 的时候,f.set(M,5)这里会出错;而将final或static去掉的时候,set(M,5)都不会出错。这就奇了怪了,于是我查了下文档,发现这句话:

If the underlying field is final, the method throws an  IllegalAccessException unless setAccessible(true) has succeeded for this Field object and the field is non-static.

也就是说static final 会使得方法抛出一个异常。


那通过反射将final字段修改掉行不行啊

于是我又改了下程序:

import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
class My{
private static final Integer value= 1;
private static String str="sds";
public static int getValue()
{
return value;
}
public String getString()
{
return str;
}
}
public class FinalReflect{
public static void main(String[] args) throws Exception
{
Class c = Class.forName("reflect.My");
My M = (My)c.newInstance();
Field f = c.getDeclaredField("value");
Field s=c.getDeclaredField("str");
s.setAccessible(true);
f.setAccessible(true);
<span style="white-space:pre">	</span>
Field modifiersField = Field.class.getDeclaredField("modifiers");
modifiersField.setAccessible(true);
System.out.println("Modifier before modify: "+f+" "+f.getModifiers());
modifiersField.setInt(f, f.getModifiers() & ~Modifier.FINAL);
System.out.println("Modifier after modify: "+f+" "+f.getModifiers());
f.set(M,5);
s.set(null ,"ss");
System.out.println("Field.get: "+f.get(M));
System.out.println("Field.get: "+s.get(M));
}
}
输出为

Modifier before modify: private static final java.lang.Integer reflect.My.value 26
Modifier after modify: private static java.lang.Integer reflect.My.value 10
Field.get: 5
Field.get: ss


我们看到打印出来的前两行,final 真的消失了,value的修饰符被反射修改了   

再看一下Modifier.class中定义的  public static final int FINAL = 0x00000010; 确实是减少的16;

那么剩下的值代表什么呢?

对了,就是static和private

public static final int STATIC  = 0x00000008;

public static final int PRIVAT   = 0x00000002;


到这里终于知道为啥对于static final修饰的变量  反射的时候需要用到Modifier了;而只是final修饰的时候其实并不一定需要。

这里还有一个小知识点,我之前看到网上许多人在使用set()的时候第一个参数用null 然后我就懵了,为啥我用就出错?看了文档才发现

If the underlying field is static, the obj argument is ignored; it may be null.  人家是针对static的,我用的使用写的是final  。。。。


血与泪告诉我们看文档的重要性。特别像是我这样的小白!




最后

以上就是坚强凉面为你收集整理的对java Field中的set()方法以及Modifier的一些思考的全部内容,希望文章能够帮你解决对java Field中的set()方法以及Modifier的一些思考所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(55)

评论列表共有 0 条评论

立即
投稿
返回
顶部