简单介绍Java语言中内存管理的几个技巧

2016-02-19 20:07 2 1 收藏

岁数大了,QQ也不闪了,微信也不响了,电话也不来了,但是图老师依旧坚持为大家推荐最精彩的内容,下面为大家精心准备的简单介绍Java语言中内存管理的几个技巧,希望大家看完后能赶快学习起来。

【 tulaoshi.com - 编程语言 】

  Java做的系统给人的印象是什么?占内存!说道这句话就会有N多人站出来为java辩护,并举出一堆的性能测试报告来证实这一点。

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

  其实从理论上来讲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 TRUE = new Boolean(true);

  public static final Boolean FALSE = new Boolean(false);

  需要的时候只要取这两个变量就可以了,

  比如:

  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(比如IBM SDK)中使用的话,那么可以自己做了工具类封装一下,比如IntegerUtils.valueOf(),这样就可以在任何SDK中都可以使用这种特性。

  (3)用StringBuffer代替字符串相加。这个我就不多讲了,因为已经被人讲过N次了。我只想将一个不是笑话的笑话,我在看国内某“闻名”java开发的WEB系统的源码中,竟然发现其中大量的使用字符串相加,一个拼装SQL语句的方法中竟然最多构造了将近100个string实例。无语中!

  (4)过滥使用哈希表,有一定开发经验的开发人员经常会使用hash表(hash表在JDK中的一个实现就是HashMap)来缓存一些数据,从而提高系统的运行速度。比如使用HashMap缓存一些物料信息、人员信息等基础资料,这在提高系统速度的同时也加大了系统的内存占用,非凡是当缓存的资料比较多的时候。其实我们可以使用操作系统中的缓存的概念来解决这个问题,也就是给被缓存的分配一个一定大小的缓存容器,按照一定的算法淘汰不需要继续缓存的对象,这样一方面会因为进行了对象缓存而提高了系统的运行效率,同时由于缓存容器不是无限制扩大,从而也减少了系统的内存占用。现在有很多开源的缓存实现项目,比如ehcache、oscache等,这些项目都实现了FIFO、MRU等常见的缓存算法。

  (5)避免过深的类层次结构和过深的方法调用。因为这两者都是非常占用内存的(非凡是方法调用更是堆栈空间的消耗大户)。

  (6)变量只有在用到它的时候才定义和实例化。

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

  (7)尽量避免使用static变量,类内私有常量可以用final来代替。


来源:https://www.tulaoshi.com/n/20160219/1622795.html

延伸阅读
C语言的位字段是个比较有意思的特性。它的目的是在一个机器字中保存多个对象(每个对象占据若干bit),从而节省内存资源,同时又避免复杂的位运算。在此不再讨论位字段的具体语法,下面将研究位字段的存储特性。 先说含有多个字段(field)的字(word)所占空间的规律——含有多个字段的字的大小是所有字段的类型中的最长的那个的倍数。但要...
提起Java内部类(Inner Class)可能很多人不太熟悉,实际上类似的概念在C++里也有,那就是嵌套类(Nested Class),关于这两者的区别与联系,在下文中会有对比。内部类从表面上看,就是在类中又定义了一个类(下文会看到,内部类可以在很多地方定义),而实际上并没有那么简单,乍看上去内部类似乎有些多余,它的用处对于初学者来说可能并不是那...
Java提供了两类主要的异常:runtime exception和checked exception。所有的checked exception是从java.lang.Exception类衍生出来的,而runtime exception则是从java.lang.RuntimeException或java.lang.Error类衍生出来的。 !-- frame contents -- !-- /frame contents -- 它们的不同之处表现在两方面:机制上和逻...
一、Java内存回收机制 不论哪种语言的内存分配方式,都需要返回所分配内存的真实地址,也就是返回一个指针到内存块的首地址。Java中对象是采用new或者反射的方法创建的,这些对象的创建都是在堆(Heap)中分配的,所有对象的回收都是由Java虚拟机通过垃圾回收机制完成的。GC为了能够正确释放对象,会监控每个对象的运行状况,对他们的申请、引...
  什么是UDP协议 UDP协议的全称是用户数据报,在网络中它与TCP协议一样用于处理数据包。在OSI模型中,在第四层——传输层,处于IP协议的上一层。UDP有不提供数据报分组、组装和不能对数据包的排序的缺点,也就是说,当报文发送之后,是无法得知其是否安全完整到达的。 为什么要使用UDP 在选择使用协议的时候,选择UDP必须要谨慎。在...

经验教程

804

收藏

43
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部