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);這結果改了,但是改的代碼太多。、
我想只能改change方法。也不需要返回值,來解決問題。先看了下
代碼如下:
我想只能改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 那么只要改掉
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屬性,。于是又了改了這個值代碼如下
我又試了下
改別的代碼里面的值 ,所以這只是一個簡單的例子。只是驗證通過反射能做一些讓你無法想象的東西。。
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();
}
}
發現能解決問題。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 發現一樣可以
代碼如下:
個人覺得只要Field.setAccessible(true); 之后,即使是final關鍵字標示過得屬性也可以有訪問權限!這樣的反射會改變JAVA的結構,甚至你的代碼可維護性,你完全可以代碼如下:
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();
}
}
改別的代碼里面的值 ,所以這只是一個簡單的例子。只是驗證通過反射能做一些讓你無法想象的東西。。