青菜猫(孙宇博客),青菜猫(孙宇博客),青菜猫(孙宇博客)http://www.javasdc.cn/
posts - 29,  comments - 63,  trackbacks - 0
这几天闲来无事。在网上看了一个题目,相信大家都知道这个题目
 static void change(String str){
        str="welcome";
    }
     
    public static void main(String[] args) {
        String str = "123";
        change(str);
        System.out.println(str);
    }
str =多少?
如果大家都知道 。。
然后我想把最后输出要“welcome”怎么做?
于是改成这样
  static String change(String str){
        str="welcome";
        return str;
    }
然后main里
str=change(str);这结果改了,但是改的代码太多。、
我想只能改change方法。也不需要返回值,来解决问题。先看了下
String类发现 private final char value[]; 是私有的。把这个值改了不就OK了嘛
于是想到了反射:
代码如下:
static void change(String str){
        try {
            Class<?> clazz = str.getClass();
            Field fields = clazz.getDeclaredField("value");
            Object obj = fields.get(str);
            char [] charValue = (char [] )obj;
            System.out.println(charValue);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
发现打出结果 welcome 那么只要改掉
charValue就OK了吧。于是继续加代码 发现 modifiers "private final" 会出这个错,
要加上
    field.setAccessible(true);
  try {
            Class<?> clazz = str.getClass();
            Field field = clazz.getDeclaredField("value");
            field.setAccessible(true);
            Object obj = field.get(str);
            char [] charValue = (char [] )obj;
            charValue = new char [3];
            for(int i=0;i<charValue.length;i++){
            charValue[i]='a';
            }
            field.set(str, charValue);
        } catch (Exception e) {
            e.printStackTrace();
        }
}发现能满足我的要求,。值是被改了,然后我想改变长度,于是又动一下代码charValue = new char [4];
发现值不能改变 继续看String源码,发现有个count属性,。于是又了改了这个值代码如下

 try {
            Class<?> clazz = str.getClass();
            Field field = clazz.getDeclaredField("value");
            field.setAccessible(true);
            Object obj = field.get(str);
            char [] charValue = (char [] )obj;
            charValue = new char [4];
            for(int i=0;i<charValue.length;i++){
                charValue[i]='a';
            }
             Field field1 = clazz.getDeclaredField("count");
             field1.setAccessible(true);
             field1.set(str, charValue.length);
             field.set(str, charValue);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
发现能解决问题。
我又试了下
然后我又试下Integer 发现一样可以
代码如下:
static void changeInteger(Integer a){   
        try {
            Class<?> clazz = a.getClass();
            Field field = clazz.getDeclaredField("value");
            field.setAccessible(true);
            field.set(a, 12312);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
个人觉得只要Field.setAccessible(true); 之后,即使是final关键字标示过得属性也可以有访问权限!这样的反射会改变JAVA的结构,甚至你的代码可维护性,你完全可以
改别的代码里面的值 ,所以这只是一个简单的例子。只是验证通过反射能做一些让你无法想象的东西。。

posted on 2013-06-08 11:56 青菜猫(孙宇) 阅读(20436) 评论(6)  编辑  收藏 所属分类: java


FeedBack:
# re: java反射 修改静态方法的值 setAccessible太无节操了
2013-06-09 08:20 | 开发吧
写的一针见血,精品  回复  更多评论
  
# re: java反射 修改静态方法的值 setAccessible太无节操了
2013-06-25 13:19 | ghyghost
节操?何谓节操?求解释  回复  更多评论
  
# re: java反射 修改静态方法的值 setAccessible太无节操了
2014-01-18 15:19 | zzy8811
jvm 加上安全管理期及相关策略即可防止。  回复  更多评论
  
# re: java反射 修改静态方法的值 setAccessible太无节操了
2014-03-01 03:45 | BBB
这么些会个屁的  回复  更多评论
  
# re: java反射 修改静态方法的值 setAccessible太无节操了
2014-08-10 10:33 | 江奕铭
这文章写的好!  回复  更多评论
  
# re: java反射 修改静态方法的值 setAccessible太无节操了
2015-11-28 11:44 | 宋大傻
@江奕铭
写的真受用。见识了  回复  更多评论
  

只有注册用户登录后才能发表评论。


网站导航:
 
<2014年8月>
272829303112
3456789
10111213141516
17181920212223
24252627282930
31123456

青菜猫(孙宇)结交天下朋友,在网上吸取知识..

常用链接

留言簿(16)

随笔分类

随笔档案

文章分类

搜索

  •  

最新评论

阅读排行榜

评论排行榜

青菜猫(孙宇博客),青菜猫(孙宇博客),青菜猫(孙宇博客)http://www.javasdc.cn/