暴力反射改变不可变类:String

我们都知道String是java中一个不可变类,因为String内部是一个final修饰的char数组:

private final char value[];

由于value是私有的final变量,String中也没有提供get和set方法,使得value无法改变。

但是value是一个引用,就像c++中的指针一样,指向一个数组内存的地址,被final修饰无法改变他的引用指向,但是我们可以直接改变内存数组中的数据,value是私有的,获取就需要用到反射:

public static void change(String s) throws Exception{
    	
    	System.out.println("before: "+s);
    	Field valueField = s.getClass().getDeclaredField("value");
    	valueField.setAccessible(true);
    	
    	char[] temp = (char[])valueField.get(s);
    	temp[0] = 'L';
    	System.out.println("after: "+s);
    }

结果:

//输入abcd
before: abcd
after: Lbcd

我们使用Field的setAccessible方法使得private的变量变的可以访问,获取到数组,直接更改数组中的值,但这样破坏了类的访问规则,会产生很多安全隐患,知乎上看到一个回答:

setAccessible是一种hack

潜台词是:你清楚内部实现,你知道你在做什么,相信你不会搞砸,平时不需要过分担心“如果别人用setAccessible来搞我怎么办”之类的问题,死代码防不了活人,也没必要

当然,java将String设置为不可变是有道理的,暴力的更改会带来很多问题,例如:

        String s1 = "abcd";
	String s2 = "abcd";
	Field valueField = s1.getClass().getDeclaredField("value");
    	valueField.setAccessible(true);
    	
    	char[] temp = (char[])valueField.get(s1);
    	temp[0] = 'L';
    	System.out.println(s1);
    	System.out.println(s2);

只改变s1的值,s2也变了,由于字符串存储在方法区常量池中,s1和s2指向同一块内存区域,通过s1改变了字符串常量的值,s2也随着改变了。

所以不到万不得已,不要随便通过暴力手段更改不可变类的值,会有很多难以发现的问题存在。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值
>