语法糖(Syntactic Sugar),也叫糖衣语法,是英国计算机科学家彼得·约翰·兰达(Peter J. Landin)发明的一个术语。指的是,在计算机语言中添加某种语法,这种语法能使程序员更方便的使用语言开发程序,同时增强程序代码的可读性,避免出错的机会;但是这种语法对语言的功能并没有影响。
Java中的泛型,变长参数,自动拆箱/装箱,条件编译等都是,下面做简单的介绍和分析。
泛型
与C#中的泛型相比,Java的泛型可以算是“伪泛型”了。在C#中,不论是在程序源码中、在编译后的中间语言,还是在运行期泛型都是真实存在的。Java则不同,Java的泛型只在源代码存在,只供编辑器检查使用,编译后的字节码文件已擦除了泛型类型,同时在必要的地方插入了强制转型的代码。
泛型代码:
- public static void main(String[] args) {
- List<String> stringList = new ArrayList<String>();
- stringList.add('oliver');
- System.out.println(stringList.get(0));
- }
将上面的代码的字节码反编译后:
- public static void main(String args[])
- {
- List stringList = new ArrayList();
- stringList.add('oliver');
- System.out.println((String)stringList.get(0));
- }
自动拆箱/装箱 自动拆箱/装箱是在编译期,依据代码的语法,决定是否进行拆箱和装箱动作。
装箱过程:把基本类型用它们对应的包装类型进行包装,使基本类型具有对象特征。
拆箱过程:与装箱过程相反,把包装类型转换成基本类型。
需要注意的是:包装类型的“==”运算在没有遇到算数运算符的情况下不会自动拆箱,而其包装类型的equals()方法不会处理数据类型转换,所以:
- Integer a = 1;
- Integer b = 1;
- Long c = 1L;
- System.out.println(a == b);
- System.out.println(c.equals(a));
这样的代码应该尽量避免自动拆箱与装箱。
循环历遍(foreach) 语法:
- List<Integer> list = new ArrayList<Integer>();
- for(Integer num : list){
- System.out.println(num);
- }
Foreach要求被历遍的对象要实现Iterable接口,由此可想而知,foreach迭代也是调用底层的迭代器实现的。反编译上面源码的字节码:
- List list = new ArrayList();
- Integer num;
- Integer num;
- for (Iterator iterator = list.iterator(); iterator.hasNext(); System.out.println(num)){
- num = (Integer) iterator.next();
- }
条件编辑 很多编程语言都提供了条件编译的途径,C,C 中使用#ifdef。Java语言并没有提供这种预编译功能,但是Java也能实现预编译。
- if(true){
- System.out.println('oliver');
- }else{
- System.out.println('lee');
- }
这段代码的字节码反编译后只有一条语句:
- System.out.println('oliver');
在编译器中,将会把分支不成立的代码消除,这一动作发生在编译器解除语法糖阶段。
所以说,可以利用条件语句来实现预编译。