可变长参数
可变参数(variable arguments
)是在 Java 1.5 中引入的一个特性。它允许一个方法把任意数量的值作为参数。
看下以下可变参数代码,其中 print
方法接收可变参数:
public static void main(String[] args){print("Holis", "公众号:Hollis", "博客:www.hollischuang.com", "QQ:907607222");}public static void print(String... strs)
{for (int i = 0; i < strs.length; i++){System.out.println(strs[i]);}
}
反编译后代码:
public static void main(String args[])
{print(new String[] {"Holis", "\u516C\u4F17\u53F7:Hollis", "\u535A\u5BA2\uFF1Awww.hollischuang.com", "QQ\uFF1A907607222"});
}public static transient void print(String strs[])
{for(int i = 0; i < strs.length; i++)System.out.println(strs[i]);}
从反编译后代码可以看出,可变参数在被使用的时候,他首先会创建一个数组,数组的长度就是调用该方法是传递的实参的个数,然后再把参数值全部放到这个数组当中,然后再把这个数组作为参数传递到被调用的方法中。
枚举
Java SE5 提供了一种新的类型-Java 的枚举类型,关键字enum
可以将一组具名的值的有限集合创建为一种新的类型,而这些具名的值可以作为常规的程序组件使用,这是一种非常有用的功能。
要想看源码,首先得有一个类吧,那么枚举类型到底是什么类呢?是enum
吗?答案很明显不是,enum
就和class
一样,只是一个关键字,他并不是一个类,那么枚举是由什么类维护的呢,我们简单的写一个枚举:
public enum t {SPRING,SUMMER;
}
然后我们使用反编译,看看这段代码到底是怎么实现的,反编译后代码内容如下:
public final class T extends Enum
{private T(String s, int i){super(s, i);}public static T[] values(){T at[];int i;T at1[];System.arraycopy(at = ENUM$VALUES, 0, at1 = new T[i = at.length], 0, i);return at1;}public static T valueOf(String s){return (T)Enum.valueOf(demo/T, s);}public static final T SPRING;public static final T SUMMER;private static final T ENUM$VALUES[];static{SPRING = new T("SPRING", 0);SUMMER = new T("SUMMER", 1);ENUM$VALUES = (new T[] {SPRING, SUMMER});}
}
断言
在 Java 中,assert
关键字是从 JAVA SE 1.4 引入的,为了避免和老版本的 Java 代码中使用了assert
关键字导致错误,Java 在执行的时候默认是不启动断言检查的(这个时候,所有的断言语句都将忽略!),如果要开启断言检查,则需要用开关-enableassertions
或-ea
来开启。
public class AssertTest {public static void main(String args[]) {int a = 1;int b = 1;assert a == b;System.out.println("公众号:Hollis");assert a != b : "Hollis";System.out.println("博客:www.hollischuang.com");}
}
反编译后代码如下:
public class AssertTest {public AssertTest(){}public static void main(String args[])
{int a = 1;int b = 1;if(!$assertionsDisabled && a != b)throw new AssertionError();System.out.println("\u516C\u4F17\u53F7\uFF1AHollis");if(!$assertionsDisabled && a == b){throw new AssertionError("Hollis");} else{System.out.println("\u535A\u5BA2\uFF1Awww.hollischuang.com");return;}
}static final boolean $assertionsDisabled = !com/hollis/suguar/AssertTest.desiredAssertionStatus();}
很明显,反编译之后的代码要比我们自己的代码复杂的多。所以,使用了 assert 这个语法糖我们节省了很多代码。其实断言的底层实现就是 if 语言,如果断言结果为 true,则什么都不做,程序继续执行,如果断言结果为 false,则程序抛出 AssertError 来打断程序的执行。-enableassertions
会设置$assertionsDisabled 字段的值。
数值字面量
在 java 7 中,数值字面量,不管是整数还是浮点数,都允许在数字之间插入任意多个下划线。这些下划线不会对字面量的数值产生影响,目的就是方便阅读。
public class Test {public static void main(String... args) {int i = 10_000;System.out.println(i);}
}
反编译后:
public class Test
{public static void main(String[] args){int i = 10000;System.out.println(i);}
}
反编译后就是把_
删除了。也就是说 编译器并不认识在数字字面量中的_
,需要在编译阶段把他去掉。