由于Java中的所有对象都是句柄,并且由于每个工具都是在内存堆中建设的——只有不再需要的时候,才会看成垃圾收集掉,所以工具的操纵方法产生了变革,出格是在通报和返回工具的时候。举个例子来说,在C和C++中,假如想在一个要领里初始化一些存储空间,大概需要请求用户将那片存储区域的地点通报进入要领。不然就必需思量由谁认真排除那片区域。因此,这些要领的接口和对它们的领略就显得要巨大一些。但在Java中,基础不必体贴由谁认真排除,也不必体贴在需要一个工具的时候它是否仍然存在。因为系统会为我们顾问一切。我们的措施可在需要的时候建设一个工具。并且更进一步地,基础不必担忧谁人工具的传输机制的细节:只需简朴地通报句柄即可。有些时候,这种简化很是有代价,但另一些时候却显得有些多余。
可从两个方面认识这一机制的缺点:
(1) 必定要为特另外内存打点支付效率上的损失(尽量损失不大),并且对付运行所需的时间,老是存在一丝不确定的因素(因为在内存不足时,垃圾收集器大概会被强制采纳动作)。对大大都应用来说,利益显得比缺点重要,并且部门对时间要求很是苛刻的段落可以用native要领写成(拜见附录A)。
(2) 别名处理惩罚:有时会不慎得到指向同一个工具的两个句柄。只有在这两个句柄都假定指向一个“明晰”的工具时,才有大概发生问题。对这个问题,必需加以足够的重视。并且应该尽大概地“克隆”一个工具,以防备另一个句柄被不但愿的窜改影响。除此以外,可思量建设“不行变”工具,使它的操纵能返回同种范例或差异种范例的一个新工具,从而提高措施的执行效率。但千万不要改变原始工具,使对谁人工具别名的其他任何方面都感受不出变革。
有些人认为Java的克隆是一个鸠拙的家伙,所以他们实现了本身的克隆方案(注释⑤),永远杜绝挪用Object.clone()要领,从而消除了实现Cloneable和捕捉CloneNotSupportException违例的需要。这一做法是公道的,并且由于clone()在Java尺度库中很少得以支持,所以这显然也是一种“安详”的要领。只要不挪用Object.clone(),就不必实现Cloneable可能捕捉违例,所以那看起来也是可以或许接管的。
⑤:Doug Lea出格重视这个问题,并把这个要领推荐给了我,他说只需为每个类都建设一个名为duplicate()的函数即可。
Java中一个有趣的要害字是byvalue(按值),它属于那些“保存但未实现”的要害字之一。在领略了别名和克隆问题今后,各人可以想象byvalue最终有一天会在Java顶用于实现一种自动化的当地副本。这样做可以办理更多巨大的克隆问题,并使这种环境下的编写的代码变得越发简朴和结实。