其实《高效Java》的第2章我读了很多次,每一次都觉得很烦,因为第2章说了很多小点,可是只有两个目的:
(1)不要重复创建对象
(2)消除过期对象的引用
弄懂这两个点,我觉得第2张理解的差不多了吧。我们看看怎么实现这两点。
-
不要重复创建对象
(1)使用静态工厂方法
调用静态工厂方法,不用我们创建对象,直接调用方法。如String.valueOf()。但是并不是说,如果这么好用,方法能定义为静态方法尽量定义咯。你傻呀,是有限制的好吗?既然他是静态方法,就使用使用静态变量,不能使用对象变量。我们一般在哪里会用到呢?比如工具类啦,又比如获取单例对象的时候啦。既然在这里说到了单例,接下来看看呗!
(2)使用单例
在《高效Java》中,提到使用“私有构造函数强化不可实例化能力”,这也就是单例的思想啦。单例模式确保一个一个类只有一个实例,这个也可以避免我们重复创建对象,节省资源。但是不要什么类都使用单例哟?一般都是需要某个类只能实例化一个实例,避免多个实例引起程序逻辑错误的场合下使用。结合我自己的经验,单例还能聊一下。单例还分饥饿式和懒惰式。饥饿式呢就是在定义单例属性的时候就初始化,缺点是可能会造成浪费内存,因为你不一定使用它,但是他已经初始化了;优点呢,就是保证线程安全,在多线程环境中,不会出现2个这样的实例。懒惰式呢,就是在外部类要得到这个单例的时候,才初始化。由于不会一开始就初始化,所以不会造成内存浪费;缺点呢,你要做好多线程安全处理。具体自己百度去吧,有很多解决方案。
(3)适当的new,重用对象
在《高效Java》提到一个例子:
String s = new String("silly"); // Don't do this
String s = "No longer sliiy"; // Should do this
这样只是使用了一个String实例,而不是创建一个新的实例。听我说,String这个类比较特殊,JVM在处理字符串的时候会采用缓存的策略,也就是说,如果一个String的值在编译期间就可以确定,那么系统会将这个字符串存储到常量池当中,下次如果有其他的String对象的值和之前出现的值相等,那么系统会自动将这个对象指向常量池里对应的字符串,从而节约空间。这也是String的高效用法。其实这个是重用的思想,还有我们ListView,如果要做到节省内存,就可以用到ViewHolder。啥是ViewHolder,问百度。 -
消除过期对象的引用
因为Java有GC帮我们管理着内存,所以我们new完一个对象,也就不用管它死活了。其实不然。如果我们new完一个对象,如果实例没有引用了到它,那它就属于一个GC可到达的地方,GC就可以清理他。可是遇到这种情况:
class StackA {
public String[] ss = new String[10];
public int size = 0;
public void push(String s) {
if(size < 10) {
ss[size++] = s;
}
}
public String pop() {
if(size == 0) return;
return ss[size--]; // 内存泄露的地方
}
}
我们看到泄露的地方,虽然size是--了,但是ss[size]还是存在的,数组还引用到它,但是并不使用,这就相当于ss[size]已经是过期的对象了,我们就应该把他消除,怎么消除?
我们重写这个pop这个方法,代码如下:
public String pop() {
if(size == 0) return;
ss[size] = null;
return ss[size--];
}
就这么简单我们就消除了过期的引用。因为这是一个经典的例子,所以大家都应该了解一下过期引用是个什么东东,应该怎么消除。主要的思想还是,如果你不用了它(对象),就不要“占有”它(对象)。这样就很好可以消除过期引用了。
还有一种就是使用软引用和弱引用技术,使用软引用保存的对象,也可以成为GC回收的对象。至Android 2.3之后,GC更倾向于回收软引用和弱引用,使用的时候注意判断null。关于软引用和弱引用的使用,大家还是百度吧!!
以上是个人总结出来的,如有不对,欢迎指正!!!
网友评论