文章目錄
- 概覽
- 泛型
- 增強for循環
- 自動裝箱與拆箱
- 字符串拼接
- 枚舉類型
- 可變參數
- 內部類
- try-with-resources
- Lambda表達式
概覽
語法糖是指編程語言中的一種語法結構,它們并不提供新的功能,而是為了讓代碼更易讀、更易寫而設計的。語法糖使得某些常見的編程模式或操作變得更加簡潔和直觀,但在底層實現上,并沒有引入新的語言特性或改變語言的表達能力。
盡管語法糖使得代碼更加簡潔和易讀,但實際執行時,底層代碼仍然需要遵循編程語言的基本語法和規則。在編譯階段,編譯器將使用語法糖編寫的高級語法轉換為更基礎、更原始的語法結構,便于生成相應的目標代碼,如字節碼或機器碼,這就是解語法糖的過程。這個過程確保了最終執行的代碼在語義上與原始的語法糖一致,同時能夠在目標環境中正確運行。Java編譯器的desugar()
方法負責這個過程,這個過程確保了Java語言的高級特性可以在不增加JVM復雜性的情況下實現,從而提高了開發效率和代碼可讀性。
舉例來說,假設有如下的Java語法糖:
List<String> list = new ArrayList<>();
list.add("Hello");
String s = list.get(0);
在這段代碼中,泛型<String>
是語法糖。編譯器在編譯這段代碼時,會將其轉換為如下的基礎語法:
List list = new ArrayList();
list.add("Hello");
String s = (String) list.get(0);
泛型
泛型允許類、接口和方法在聲明時使用參數化類型,提供了編譯時類型安全檢查機制,避免了強制類型轉換的麻煩。但Java中的泛型只在程序源代碼中有效,在編譯后的字節碼中會自動用強制類型轉換進行替代。也就是說,Java語言中的泛型機制其實就是一顆語法糖,
// 泛型示例
List<String> names = new ArrayList<>();
names.add("Alice");
String first = names.get(0);
// 解語法糖
List names = new ArrayList();
names.add("Alice");
String first = (String) names.get(0);
增強for循環
增強for循環用于遍歷數組或集合,即要么是一個數組,要么實現了Iterable
接口,與普通for循環相比,功能更強并且代碼更簡潔。
public static void main(String[] args) {String[] params = new String[]{"hello","world"};//增強for循環對象為數組for(String str : params){System.out.println(str);}List<String> lists = Arrays.asList("hello","world");//增強for循環對象實現Iterable接口for(String str : lists){System.out.println(str);}
}
// 解語法糖
public static void main(String[] args) {String[] params = new String[]{"hello", "world"};String[] lists = params;int var3 = params.length;//數組形式的增強for退化為普通forfor(int str = 0; str < var3; ++str) {String str1 = lists[str];System.out.println(str1);}List var6 = Arrays.asList(new String[]{"hello", "world"});Iterator var7 = var6.iterator();//實現Iterable接口的增強for使用iterator接口進行遍歷while(var7.hasNext()) {String var8 = (String)var7.next();System.out.println(var8);}}
自動裝箱與拆箱
自動裝箱和自動拆箱是Java中的語法糖,用于簡化基本數據類型和其對應包裝類型之間的轉換操作。當將基本數據類型賦值給對應的包裝類型時,編譯器會調用包裝類型的valueOf()
方法來創建一個包裝對象,并將基本數據類型的值傳遞給這個方法。當需要使用包裝類型對象中的值進行基本數據類型的操作時,編譯器會自動調用包裝類型對象的xxxValue()
方法,將包裝對象轉換為對應的基本數據類型值。
Integer boxedNum = 10; // 自動裝箱
int num = boxedNum; // 自動拆箱
// 解語法糖
Integer boxedNum = Integer.valueOf(10);
int num = boxedNum.intValue();
字符串拼接
拼接字符串最簡單的方式就是直接使用符號"+“來拼接,其實“+”是Java提供的一個語法糖。字符串拼接使用”+"操作符,在編譯時會被轉換為StringBuilder
操作。
String message = "Hello, " + name + "!";
// 解語法糖
StringBuilder sb = new StringBuilder();
sb.append("Hello, ");
sb.append(name);
sb.append("!");
String message = sb.toString();
枚舉類型
枚舉類型就是一些具有相同特性的類常量,在Java中類的定義使用class
,枚舉類的定義使用enum
。但在Java的字節碼結構中,其實并沒有枚舉類型,枚舉類型只是一個語法糖,在編譯完成后被編譯成一個普通的類。這個類繼承java.lang.Enum
,并被final
關鍵字修飾。
public enum Day {MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, SUNDAY
}
// 解語法糖
public final class Day extends Enum<Day> {public static final Day MONDAY = new Day("MONDAY", 0);public static final Day TUESDAY = new Day("TUESDAY", 1);public static final Day WEDNESDAY = new Day("WEDNESDAY", 2);public static final Day THURSDAY = new Day("THURSDAY", 3);public static final Day FRIDAY = new Day("FRIDAY", 4);public static final Day SATURDAY = new Day("SATURDAY", 5);public static final Day SUNDAY = new Day("SUNDAY", 6);private final String name;private final int ordinal;private Day(String name, int ordinal) {this.name = name;this.ordinal = ordinal;}public String name() {return name;}public int ordinal() {return ordinal;}public static Day[] values() {return (Day[]) $VALUES.clone();}public static Day valueOf(String name) {return (Day) Enum.valueOf(Day.class, name);}private static final Day[] $VALUES = {MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, SUNDAY};
}
可變參數
可變參數就是允許方法接受可變數量的參數。使用變長參數有兩個條件,一是變長的那一部分參數具有相同的類型,二是變長參數必須位于方法參數列表的最后面。變長參數同樣是Java中的語法糖,其內部實現是Java數組。
public void printNumbers(int... numbers) {}
// 解語法糖
public void printNumbers(int[] numbers) {}
內部類
內部類就是定義在一個類內部的類,之所以引入內部類是因為有些時候一個類只在另一個類中引用,我們不想讓其在另外一個地方被使用。內部類可以在一個類內部定義,但在編譯時會被轉換為獨立的類文件,并不是真正套在一個類的內部,而是分成兩個類編譯。
class Outer {class Inner {void display() {System.out.println("Inner class method");}}
}
// 解語法糖
class Outer {class Inner {final Outer outer;Inner(Outer outer) {this.outer = outer;}void display() {System.out.println("Inner class method");}}
}
try-with-resources
try-with-resources
語句簡化了資源管理,使得資源在使用后自動關閉,這個語法糖就能讓代碼及其簡潔。原理是編譯器把它轉換成了try-catch-finally
。
try (BufferedReader reader = new BufferedReader(new FileReader("file.txt"))) {String line = reader.readLine();System.out.println(line);
} catch (IOException e) {e.printStackTrace();
}
// 解語法糖
BufferedReader reader = null;
try {reader = new BufferedReader(new FileReader("file.txt"));String line = reader.readLine();System.out.println(line);
} catch (IOException e) {e.printStackTrace();
} finally {if (reader != null) {try {reader.close();} catch (IOException ex) {ex.printStackTrace();}}
}
Lambda表達式
Lambda
表達式是Java中的一種語法糖,它提供了一種簡潔地表示匿名函數的方法,在語法上的簡潔性大大提升了代碼的可讀性和編寫效率。在編譯后,Lambda
表達式會被轉換為相應的匿名內部類形式。
Runnable r = () -> System.out.println("Hello, World!");
// 解語法糖
Runnable r = new Runnable() {@Overridepublic void run() {System.out.println("Hello, World!");}
};