Java方法中的可变参数类型是一个非常重要的概念,有着非常广泛的应用。而可变参数和数组参数又有着很大的联系,下面我们看看这两者的对比以及平时如何使用。
下面是可变参数的一个简单案例:
public class TestVarArgus {
public static void dealArray(int... intArray) {
for (int i : intArray) {
System.out.println(i);
}
}
public static void main(String args[]) {
dealArray(1, 2, 3);
}
}
可变参数使用三个点 来表示,那么他怎么使用呢,其实这里我们可以传入任意长度的数组,测试代码如下:
public class TestVarArgus {
public static void dealArray(int... intArray) {
for (int i : intArray) {
System.out.print(i+" ");
}
System.out.println();
}
public static void normalArray(int[] intArray) {
for (int i : intArray) {
System.out.println(i);
}
}
public static void main(String args[]) {
dealArray();
dealArray(1);
dealArray(1, 2);
dealArray(1, 2, 3);
}
}
执行结果如下:
1
1 2
1 2 3
通过main方法里的调用,可以看出来这个可变参数既可以是没有参数(空参数),也可以是不定长的。看到这里估计都能明白,这个不定长的参数其实和数组参数挺像的。事实上,也确实是这么回事儿。编译器会在悄悄地把这最后一个形参转化为一个数组形参。请看下面idea打开的class文件代码:
// class文件
public class TestVarArgus {
public TestVarArgus() {
}
public static void dealArray(int... intArray) {
int[] var1 = intArray;
int var2 = intArray.length;
for(int var3 = 0; var3 < var2; ++var3) {
int i = var1[var3];
System.out.print(i + " ");
}
System.out.println();
}
public static void main(String[] args) {
dealArray();
dealArray(1);
dealArray(1, 2);
dealArray(1, 2, 3);
}
}
说到这里,那么可以来验证一下,看看是不是这个可变参数就是数组类参数?看代码:
public class TestVarArgus {
public static void dealArray(int... intArray) {
for (int i : intArray) {
System.out.print(i+" ");
}
System.out.println();
}
public static void dealArray(int[] intArray) {
// 会有Duplicate method dealArray(int[]) in type TestVarArgus的错误
for (int i : intArray)
System.out.print(i + " ");
System.out.println();
}
public static void main(String args[]) {
dealArray();
dealArray(1);
dealArray(1, 2);
dealArray(1, 2, 3);
}
}
从上面这段代码编译器会直接报错方法已经定义了,这两个方法是冲突的,是无法重载的。
到这里,再来做一个有意思的实验:
代码1,正常编译运行:
public class TestVarArgus {
public static void dealArray(int... intArray) {
for (int i : intArray)
System.out.print(i + " ");
System.out.println();
}
public static void main(String args[]) {
int[] intArray = { 1, 2, 3 };
dealArray(intArray);// 通过编译,正常运行
}
}
代码2,编译报错:
public class TestVarArgus {
public static void dealArray(int[] intArray) {
for (int i : intArray)
System.out.print(i + " ");
System.out.println();
}
public static void main(String args[]) {
dealArray(1, 2, 3);// 编译错误
}
}
从上面这两段代码可以看出来,可变参数是兼容数组类参数的,但是数组类参数却无法兼容可变参数。其实对于第二段代码而言,编译器并不知道什么可变不可变,在它看来,需要定义一个dealArray(int, int, int)类的方法。所以,自然就无法去匹配数组类参数的dealArray方法了。
既然Java方法接收可变参数,那么接下来我们再来看一下下面的代码:
public class TestVarArgus {
public static void dealArray(int count, int... intArray) {
}
public static void dealArray(int... intArray, int count) {
// 编译报错,可变参数类型应该作为参数列表的最后一项
}
public static void main(String args[]) {
}
}
这段代码说明了,可变参数类型必须作为参数列表的最后一项,而不能放在定长参数的前面。估计你会想到一个词“优先级”。因为没有确切的说明,只是这样一种规定,这里可以借用“优先级”这个词来理解一下,请看下面的代码:
public class TestVarArgus {
public static void dealArray(int... intArray) {
System.out.println("1");
}
public static void dealArray(int count, int count2) {
System.out.println("2");
}
public static void main(String args[]) {
dealArray(1, 2);
}
代码贴出来估计都知道是输出2,而不是1。这里需要记住:能匹配定长的方法,那么优先匹配该方法。含有不定参数的那个重载方法是最后被选中的。