《Effective Java》学习笔记(1)
类别: JAVA教程
这本书对于java程序员的意义就如《Effective C++》对于C++程序员的意义一样,我想是每个java爱好者的必读书之一了,最近在啃这本书,一些学习笔记希望能对大家有所帮助。
一。创建和销毁对象
第一条:考虑用静态工厂方法代替构造函数
实例代码 :Boolean类中的valueOf()方法
public static Boolean valueOf(boolean b)
{
return (b?Boolean.TRUE:Boolean.FALSE);
}
优点:
1。和构造函数不同,静态工厂方法有自己的名字。如果存在着多种版本的构造函数,有的仅仅是参数顺便的不同,此时你应该考虑用静态工厂方法。
2。静态工厂方法不要求一定要创建对象。可使用预先构造好的对象。例如Boolean.valueOf()方法就从不创建对象。在需要频繁创建对象,并且创建对象成本较高的情况下,你应该考虑采用静态工厂方法
3。与构造函数不同,静态工厂方法可以返回一个原返回类型的子类型的对象。这方面的最好的例子就是Collections Framework。Collections Framework有20个实用的集合接口实现,这些实现大多数是通过一个不可实例子化的java.utl.Collections中的静态工厂方法导出的。
缺点:
1。类如果不含有公有或者受保护的构造函数,就不能被继承。某种意义上这也限制了继承的滥用
2。静态工厂方法和其他静态方法一样,一般要在API文档中作出特别的说明。在没有强烈的需要下,你还是应该使用规范的构造函数。
第2条:使用私有构造函数强化singleton属性
所谓singleton是指这样的类,它只能被实例化一次/(也就是单例模式),有两种方式,如下:
1。提供一个静态常量
public class Example{
public static final Example INSTANCE=new Example();
private Example(){ //构造函数为私有
...}
....
}
2。使用静态工厂方法
public class Example{
private static final Example INSTANCE=new Example();//改为私有
private Example(){ //构造函数为私有
...}
public static Example getInstance(){
return INSTANCE;
}
....
}
第一种方法在性能上可能更好,第2种方法提供了更大的灵活性,你可以决定是否做成singleton。要使一个singleton的类变成可序列化的,仅仅实现Serializable接口是不够,还必须提供一个readResolve()方法,否则会产生一个新的实例。违背了singleton的本意
private Object readResolve() throws ObjectStreamException{
return INSTANCE;
}
第3条。通过私有构造函数强化不可实例能力
也就是不使某个类不能产生任何对象。或者你要说写成抽象类不就可以了?NO,抽象类可以被实现,其子类也可以被实现。我们要的是绝对不能被实例化的类,这种类一般只有一些静态变量和静态方法,只是作为工具类使用,如java.utl.Arrays。要做到这一点只要包含一个私有的显式构造函数。这样同时也保证了这个类不能被继承,因为子类无法访问父类的构造函数。
第4条:避免重复创建对象
如果一个对象是非可变的,那么它总可以被重用,而不是再去创建一个对象。例如
String s=new String("denny");
里面的"denny"本身就是一个实例。而这句话每次又重新创建一个同样的实例。这完全是没有必要的,如果在一个频繁调用的方法中使用这样的语句,性能上会有很大影响。应该用
String s="denny";来代替上面的语句。一个常用的方法是把重复需要用到的对象做成类的私有的静态常量(当然,要保证这些变量在创建以后不再改变),用一个static块包含他们。另外,不要以为创建对象是代价非常昂贵,相反,一些小对象的构造函数往往只做很少的工作,所以小对象的创建是非常廉价的,只有重量级的对象(如数据库连接)才需要采用对象池来重用对象。
第5条:消除过期引用
“内存泄露”!什么,我有没有听错,java也有“内存泄露”。是的,那不是C++的专利。看下面的例子
public Class Stack{
private Object[] elements;
private in size=0;
public Stack(int initialCapacity){
this.elements=new Object[initialCapacity];
}
public Object pop()
{
if(size==0)
throw new EmptyStackException();
return elements[--size];
}
....
} 这个程序并没有很明显的错误,但是随着不断增加的内存占用,程序的性能的降低会逐渐显现。原因在于这个栈收缩的时候,从栈中弹出的对象并不会被当作垃圾回收,这是因为栈内部维持着这些对象的过期引用,也就是永远也不会再被解除的引用,应该把pop操作修改下:
public Object pop()
{
if(size==0)
throw new EmptyStackException();
Object result=elements[--size];
elements[size]==null; //把引用设为null
return result;
}
自己管理内存的类一般都存在着这样的问题,必须时刻警惕。内存泄露的另一个来源就是缓存了,你缓存了一个对象,却忘记了去释放。内存泄露问题可以通过专门的工具来检测。
第6条:避免使用终结函数(finalize())
想起一次在CSDN论坛上,有人问什么时候该使用finlize(),和C++有什么不同,我竟然回答说可以在finalize()方法中处理一些关闭资源的操作(关闭文件等等)。汗颜!终结函数并不能保证会被及时地执行,从一个对象变的不可到达(通过对象网络没有了这个对象的引用),到它的终结函数被执行,这段时间的长短是任意的,不确定的。所以,时间关键的任务不应该由终结函数来完成,例如关闭一个已经被打开的文件。由于JVM延迟执行终结函数,所以大量的文件保留在打开状态!而且终结函数的实现是不同的JVM中有不同的方法,所以你不能保证此函数的移植性。记住这点:
我们不应该依赖一个终结函数来更新关键性的永久状态。
那么我们该如何编程序来执行清理工作,通常提供一个显式的终止方法,通常与tr..finally结构结合使用,这方面的例子最好的是java.io里面的各种流操作了,基本都有一个close()方法,你必须显式地关闭打开的资源。终结函数的使用有两个合理的方面:
1。充当最后一道“安全网”,在客户端忘记或者不能调用显式终止方法的时候。
2。调用本地对象的时候,本地对象不拥有关键性资源的前提下,终止方法完成必要的工作以释放资源。
- 上一篇: DODS学习日记(四)
- 下一篇: 系统界面自动锁定功能实现原理
-= 资 源 教 程 =-
文 章 搜 索