1700440690
1700440691
//分别传递int类型和Integer类型
1700440692
1700440693
cilent.f(i);
1700440694
1700440695
cilent.f(Integer.valueOf(i));
1700440696
1700440697
}
1700440698
1700440699
public void f(long a){
1700440700
1700440701
System.out.println(“基本类型的方法被调用”);
1700440702
1700440703
}
1700440704
1700440705
public void f(Long a){
1700440706
1700440707
System.out.println(“包装类型的方法被调用”);
1700440708
1700440709
}
1700440710
1700440711
}
1700440712
1700440713
在上面的程序中首先声明了一个int变量i,然后加宽转变成long型,再调用f()方法,分别传递int和long的基本类型和包装类型,诸位想想该程序是否能够编译?如果能编译输出结果又是什么呢?
1700440714
1700440715
首先,这段程序绝对是能够编译的。不过,说不能编译的同学还是很动了一番脑筋的,只是还欠缺点火候,你可能会猜测以下这些地方不能编译:
1700440716
1700440717
f()方法重载有问题。定义的两个f()方法实现了重载,一个形参是基本类型,一个形参是包装类型,这类重载很正常。虽然基本类型和包装类型有自动装箱、自动拆箱的功能,但并不影响它们的重载,自动拆箱(装箱)只有在赋值时才会发生,和重载没有关系。
1700440718
1700440719
cilent. f(i)报错。i是int类型,传递到fun(long l)是没有任何问题的,编译器会自动把i的类型加宽,并将其转变为long型,这是基本类型的转换规则,也没有任何问题。
1700440720
1700440721
cilent. f(Integer.valueOf(i))报错。代码中没有f(Integer i)方法,不可能接收一个Integer类型的参数,而且Integer和Long两个包装类型是兄弟关系,不是继承关系,那就是说肯定编译失败了?不,编译是成功的,稍后再解释为什么这里编译成功。
1700440722
1700440723
既然编译通过了,我们来看一下输出:
1700440724
1700440725
基本类型的方法被调用
1700440726
1700440727
基本类型的方法被调用
1700440728
1700440729
cilent. f(i)的输出是正常的,我们已经解释过了。那第二个输出就让人很困惑了,为什么会调用f(long a)方法呢?这是因为自动装箱有一个重要的原则:基本类型可以先加宽,再转变成宽类型的包装类型,但不能直接转变成宽类型的包装类型。这句话比较拗口,简单地说就是,int可以加宽转变成long,然后再转变成Long对象,但不能直接转变成包装类型,注意这里指的都是自动转换,不是通过构造函数生成。为了解释这个原则,我们再来看一个例子:
1700440730
1700440731
public class Client{
1700440732
1700440733
public static void main(String[]args){
1700440734
1700440735
int i=100;
1700440736
1700440737
f(i);
1700440738
1700440739
}
[
上一页 ]
[ :1.70044069e+09 ]
[
下一页 ]