JAVA語(yǔ)法糖“+”運(yùn)算符
JAVA提供的“+”運(yùn)算符,如Iteger+String,從C++的角度來(lái)看總是想找到JAVA是怎么重載這個(gè)“+”運(yùn)算符,于是進(jìn)去String這個(gè)類(lèi)中看,然而并沒(méi)有什么卵發(fā)現(xiàn),于是乎想著JAVA是怎么做到的?下面來(lái)為你逐步分析下JAVA是怎么實(shí)現(xiàn)“+操作符重載的”。
示例
- public class Example {
- public static void main(String[] args) {
- Integer a = null;
- String b = a + "456";
- System.out.println(b);
- }
- }
這個(gè)程序很簡(jiǎn)單就是一個(gè)Integer和String的“+”運(yùn)算表達(dá)式。運(yùn)行結(jié)果:null456
反編譯示例程序
命令:
- javap -c Example
反編譯后的結(jié)果如下:
- Compiled from "Example.java"
- public class com.boyu.budmw.test.Example extends java.lang.Object{
- public com.boyu.budmw.test.Example();
- Code:
- 0: aload_0
- 1: invokespecial #1; //Method java/lang/Object."<init>":()V
- 4: return
- public static void main(java.lang.String[]);
- Code:
- 0: aconst_null
- 1: astore_1
- 2: new #2; //class java/lang/StringBuilder
- 5: dup
- 6: invokespecial #3; //Method java/lang/StringBuilder."<init>":()V
- 9: aload_1
- 10: invokevirtual #4; //Method java/lang/StringBuilder.append:(Ljava/lang/Object;)Ljava/lang/StringBuilder;
- 13: ldc #5; //String 456
- 15: invokevirtual #6; //Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
- 18: invokevirtual #7; //Method java/lang/StringBuilder.toString:()Ljava/lang/String;
- 21: astore_2
- 22: getstatic #8; //Field java/lang/System.out:Ljava/io/PrintStream;
- 25: aload_2
- 26: invokevirtual #9; //Method java/io/PrintStream.println:(Ljava/lang/String;)V
- 29: return
- }
我們來(lái)分析下main函數(shù)部分:
- 0:將常量null壓入操作數(shù)棧
- 1:從操作數(shù)棧中將null彈出保存到索引為1的局部變量a中
- 2:new一個(gè)StringBuilder
- 5:復(fù)制之前new出來(lái)的空間并將其壓入操作數(shù)棧
- 6:調(diào)用進(jìn)行初始化
- 9:將結(jié)果保存到操作數(shù)棧
- 10:調(diào)用StringBuilder.append(java/lang/Object)
- 13:將“456”壓入棧頂
- 15:StringBuilder.append(java/lang/String)
- 18:執(zhí)行toString函數(shù)
從上面的分析我們可以看到其最終是先生成了一個(gè)StringBuilder對(duì)象,之后的“+”操作符都是調(diào)用了StringBuilder.append()進(jìn)行“+”的。這就可以解釋上面示例程序運(yùn)行后為什么是null456了,append object的時(shí)候調(diào)用了
- public static String valueOf(Object obj) {
- return (obj == null) ? "null" : obj.toString();
- }
將object轉(zhuǎn)化為String了。
為什么JAVA不支持操作符重載
像C++中類(lèi)對(duì)操作符進(jìn)行了重載,個(gè)人覺(jué)得會(huì)操作維護(hù)難得問(wèn)題,因?yàn)椴僮鞣剌d沒(méi)有一個(gè)標(biāo)準(zhǔn)來(lái)約束大家都可以想當(dāng)然的進(jìn)行重載會(huì)造成語(yǔ)義相差大,可讀性嚴(yán)重降低,所以java中去掉操作符重載這個(gè)特性和他的高級(jí)面向?qū)ο蠛芟喾o,不糾結(jié)這個(gè)問(wèn)題。
后記
這都是在開(kāi)發(fā)過(guò)程中會(huì)經(jīng)常使用的一些東西但是可能在平時(shí)開(kāi)發(fā)過(guò)程中沒(méi)有挖的這么深入,都想當(dāng)然了,后面可以嘗試不斷挖掘這些不被發(fā)現(xiàn)的小case。