static void change(String str){
str="welcome";
}
public static void main(String[] args) {
String str = "123";
change(str);
System.out.println(str);
}
str =多少?
如果大家都知道 。。
然后我想把最后輸出要“welcome”怎么做?
于是改成這樣
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="welcome";
return str;
}
然后main里
str=change(str);這結(jié)果改了,但是改的代碼太多。、
我想只能改change方法。也不需要返回值,來解決問題。先看了下
代碼如下:
我想只能改change方法。也不需要返回值,來解決問題。先看了下
String類發(fā)現(xiàn) private final char value[]; 是私有的。把這個(gè)值改了不就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();
}
}
發(fā)現(xiàn)打出結(jié)果 welcome 那么只要改掉
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();
}
}
發(fā)現(xiàn)打出結(jié)果 welcome 那么只要改掉
charValue就OK了吧。于是繼續(xù)加代碼 發(fā)現(xiàn) modifiers "private final" 會(huì)出這個(gè)錯(cuò),
要加上
要加上
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();
}
}發(fā)現(xiàn)能滿足我的要求,。值是被改了,然后我想改變長(zhǎng)度,于是又動(dòng)一下代碼charValue = new char [4];
發(fā)現(xiàn)值不能改變 繼續(xù)看String源碼,發(fā)現(xiàn)有個(gè)count屬性,。于是又了改了這個(gè)值代碼如下
我又試了下
改別的代碼里面的值 ,所以這只是一個(gè)簡(jiǎn)單的例子。只是驗(yàn)證通過反射能做一些讓你無法想象的東西。。
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();
}
}發(fā)現(xiàn)能滿足我的要求,。值是被改了,然后我想改變長(zhǎng)度,于是又動(dòng)一下代碼charValue = new char [4];
發(fā)現(xiàn)值不能改變 繼續(xù)看String源碼,發(fā)現(xiàn)有個(gè)count屬性,。于是又了改了這個(gè)值代碼如下
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();
}
}
發(fā)現(xiàn)能解決問題。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 發(fā)現(xiàn)一樣可以
代碼如下:
個(gè)人覺得只要Field.setAccessible(true); 之后,即使是final關(guān)鍵字標(biāo)示過得屬性也可以有訪問權(quán)限!這樣的反射會(huì)改變JAVA的結(jié)構(gòu),甚至你的代碼可維護(hù)性,你完全可以代碼如下:
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();
}
}
try {
Class<?> clazz = a.getClass();
Field field = clazz.getDeclaredField("value");
field.setAccessible(true);
field.set(a, 12312);
} catch (Exception e) {
e.printStackTrace();
}
}
改別的代碼里面的值 ,所以這只是一個(gè)簡(jiǎn)單的例子。只是驗(yàn)證通過反射能做一些讓你無法想象的東西。。