飞翔的起点

从这里出发

导航

<2007年12月>
2526272829301
2345678
9101112131415
16171819202122
23242526272829
303112345

统计

常用链接

留言簿(5)

随笔分类

随笔档案

文章分类

文章档案

搜索

最新评论

阅读排行榜

评论排行榜

java重载的解析

    从昨晚开始在看java疑惑这本书,打算以后每天把看的东西以这种方式写下了
来,一方面使所学的东西得到巩固和复习,另一方面和大家共同进步.
    1. 在java中重载解析的过程是分两个阶段运行的,第一阶段选取所有可获得并且可以应用的方法或构造器,第二阶段在第一阶段选取的方法或构造器中选取最精确的一个.,如果一个方法或构造器可以接受传递给另一个方法或构造器的任何参数,那么说明第一个方法币第二个方法缺乏精确性,
    2.在设计一个类的时候,如果该类构建于另一个累的行为之上,那么你有两种选择,一种是集成,即一个累扩展另一个累;另一个是组合,即在一个类中包含另一个类的实例.
       选择的依据是:
       一个类的每一个实例都是另一个类的一个实例,还是都有另一个类的一个实例.在前一种情况下就使用集成,而在后一种情况下应该使用组合.当你不准时,优先选择组合而不是继承.
    3.静态方法
    静态方法在继承中只能被隐藏而不可以重写,静态方法只能访问静态变量,它的参数和它定义的局部变量,访问静态方法尽可能的使用类名来访问.
   今天就写到这,有问题的地方可以指正,谢谢大家.

posted on 2007-12-13 13:18 forgood 阅读(3093) 评论(3)  编辑  收藏

评论

# re: java重载的解析 2007-12-13 13:44 cc

是java解惑吧,java疑惑,难道越看越疑惑啊。  回复  更多评论   

# re: java重载的解析 2007-12-18 18:54 forgood

是呀,我也感觉如此,但是细细看的话,会发现好多经常遇到的问题,仔细的看看,可以巩固\复习一些知识,
共同进步,
  回复  更多评论   

# re: java重载的解析[未登录] 2007-12-19 09:22 ddpie

谜题46:令人混淆的构造器案例
本谜题呈现给你了两个容易令人混淆的构造器。main方法调用了一个构造器,但是它调用的到底是哪一个呢?该程序的输出取决于这个问题的答案。那么它到底会打印出什么呢?甚至它是否是合法的呢?
public class Confusing {
private Confusing(Object o) {
System.out.println("Object");
}
private Confusing(double[] dArray) {
System.out.println("double array");
}
public static void main(String[] args) {
new Confusing(null);
}
}
传递给构造器的参数是一个空的对象引用,因此,初看起来,该程序好像应该调用参数类型为Object的重载版本,并且将打印出Object。另一方面,数组也是引用类型,因此null也可以应用于类型为double[ ]的重载版本。你由此可能会得出结论:这个调用是模棱两可的,该程序应该不能编译。如果你试着去运行该程序,就会发现这些直观感觉都是不对的:该程序打印的是double array。这种行为可能显得有悖常理,但是有一个很好的理由可以解释它。
Java的重载解析过程是以两阶段运行的。第一阶段选取所有可获得并且可应用的方法或构造器。第二阶段在第一阶段选取的方法或构造器中选取最精确的一个。如果一个方法或构造器可以接受传递给另一个方法或构造器的任何参数,那么我们就说第一个方法比第二个方法缺乏精确性[JLS 15.12.2.5]。
在我们的程序中,两个构造器都是可获得并且可应用的。构造器Confusing(Object)可以接受任何传递给Confusing(double[ ])的参数,因此Confusing(Object)相对缺乏精确性。(每一个double数组都是一个Object,但是每一个Object并不一定是一个double数组。)因此,最精确的构造器就是Confusing(double[ ]),这也就解释了为什么程序会产生这样的输出。
如果你传递的是一个double[ ]类型的值,那么这种行为是有意义的;但是如果你传递的是null,这种行为就有违直觉了。理解本谜题的关键在于在测试哪一个方法或构造器最精确时,这些测试没有使用实际的参数:即出现在调用中的参数。这些参数只是被用来确定哪一个重载版本是可应用的。一旦编译器确定了哪些重载版本是可获得且可应用的,它就会选择最精确的一个重载版本,而此时使用的仅仅是形式参数:即出现在声明中的参数。
要想用一个null参数来调用Confusing(Object)构造器,你需要这样写代码:new Confusing((Object)null)。这可以确保只有Confusing(Object)是可应用的。更一般地讲,要想强制要求编译器选择一个精确的重载版本,需要将实际的参数转型为形式参数所声明的类型。
以这种方式来在多个重载版本中进行选择是相当令人不快的。在你的API中,应该确保不会让客户端走这种极端。理想状态下,你应该避免使用重载:为不同的方法取不同的名称。当然,有时候这无法实现,例如,构造器就没有名称,因而也就无法被赋予不同的名称。然而,你可以通过将构造器设置为私有的并提供公有的静态工厂,以此来缓解这个问题[EJ Item 1]。如果构造器有许多参数,你可以用Builder模式[Gamma95]来减少对重载版本的需求量。
如果你确实进行了重载,那么请确保所有的重载版本所接受的参数类型都互不兼容,这样,任何两个重载版本都不会同时是可应用的。如果做不到这一点,那么就请确保所有可应用的重载版本都具有相同的行为[EJ Item 26]。
总之,重载版本的解析可能会产生混淆。应该尽可能地避免重载,如果你必须进行重载,那么你必须遵守上述方针,以最小化这种混淆。如果一个设计糟糕的API强制你在不同的重载版本之间进行选择,那么请将实际的参数转型为你希望调用的重载版本的形式参数所具有的类型。   回复  更多评论   


只有注册用户登录后才能发表评论。


网站导航: