JAVA做的系统给人的印象是什么?占内存!说道这句话就会有N多人站出来为JAVA辩护,并举出一堆的性能测试报告来证明这一点,其实从理论上来讲JAVA做的系统并比其他语言开发出来的系统更占用内存,那么为什么却有这么N多理由来证明它确实占内存呢???两个字,陋习。。。
(1) 别用new Boolean()
在很多场景中Boolean类型时必须的。比如JDBC中boolean类型的set与get都是通过Boolean封装传递的,大部分ORM也是用Boolean来封装Boolean类型的,比如:
ps.setBoolean("isClosed",new Boolean(true));
ps.setBoolean("isClosed",new Boolean(isClosed));
ps.setBoolean("isClosed",new Boolean(i==3));
通常这些系统中构造的Boolean实例的个数是相当多的,所以系统中充满了大量Boolean实例小对象,这是相当消耗内存的。
Boolean类实际上只要两个实例就够了,一个true的实例,一个false的实例。
Boolean类提供了两个静态变量:
public static final Boolean FALSE = new Boolean(false);
public static final Boolean TRUE = new Boolean(true);
需要的时候只要取这两个变量就可以了。
ps.setBoolean("isClosed",Boolean.TRUE);
那么像2、3句那样要根据一个boolean变量来创建一个Boolean怎么办呢? 可以使用Boolean提供的静态方法:
Boolean.valueOf();
ps.setBoolean("isClosed",Boolean.valueOf(isClosed));
ps.setBoolean("isClosed",Boolean.valueOf(i==3));
因为valueOf的内存实现是:
return (b?TRUE,FALSE)
所以可以节省大量内存。
相信如果JAVA规范直接把Boolean的构造函数规定变成private,就再也不会出现这种情况了
(2)别用new Integer
和Boolean类似,java开发中使用Integer封装int 的场合非常多,并且通常用int表示的数值通常都非常小。
SUN SDK中对Integer的实例化进行了优化,Integer类缓存了-128 到 127 这256个状态的Integer,如果使用
Integer.valueOf(int i) 传入的int范围正好在此内,就返回静态实例。
这样如果我们使用Integer.valueOf代替new Integer的话也将大大降低内存的占用。如果你的系统要在不同的SDK中使用的话,那么可以自己做了工具类封装一下,比如IntegerUtils.valueOf(),这样就可以在任何SDK中都可以使用这样的特性。
(3)用StringBuffer代替字符串相加。这个我就不多讲了,因为已经被人讲过N次了,我只想将一个不是笑话的笑话,我在看国内“著名”java开发的WEB系统的源码中,竟然发现其中大量的使用字符串相加,一个拼装SQL语句的方法中竟然最多构造将近100个String实例,无语中!
(4)过滥使用哈希表,有一定开发经验的开发人员经常会使用hash表(hash表在JDK中的一个实现就是HashMap)来缓存一些数据,从而提高系统的运行速度,比如使用HashMap缓存一些物料信息,人员信息等基础资料,这在提供系统速度的同时也加大了系统的内存占用,特别是当缓存资料比较多的时候,其实我们可以使用操作系统的中的缓存的概念解决这个问题,也就是给被缓存的分配一个一定大小的缓存容器,按照一定算法淘汰不需要进行缓存的对象,这样一方面会因为进行了对象缓存而提高了系统的运行效率,同时由于缓存容器不是无限扩大,从而减少了系统的内存占用,现在很多开源的缓存实现项目,比如ehcache,oscache等,这些项目都实现了FIFO,MRU等常见的缓存算法。
(5)避免过深的类层次和过深的方法调用,因为这两者都是非常占用内存的
(6)变量只有用到它的时候才定义和实例化。
(7)尽量避免使用static变量,类内私有常量可以用final来替代。