`
ljjr13
  • 浏览: 21757 次
  • 性别: Icon_minigender_1
  • 来自: 长沙
文章分类
社区版块
存档分类
最新评论

高性能JAVA代码之_内存管理

阅读更多

虽然JAVA的GC已经替开发者完成了对内存的管理,但GC不是万能的,很多人写的工程里依然出现OutOfMemory,所以如果JAVA的开发人员不注意内存的使用,依然会造成较高的内存消耗,从而性能很低。更甚者你写的代码,GC根本就回收不了,直接系统挂掉。GC是一段程序,不是智能,他只回收他认为的垃圾,而不是回收你认为的垃圾。

GC垃圾回收:
      Grabage Collection相信学过JAVA的人都知道这个是什么意思.但是他是如何工作的呢?
      首先,JVM在管理内存的时候对于变量的管理总是分新对象和老对象。新对象也就是开发者new出来的对象,但是由于生命周期短,那么他占用的内存并不是马上释放,而是被标记为老对象,这个时候该对象还是要存在一段时间。然后由JVM决定他是否是垃圾对象,并进行回收。
      所以我们可以知道,垃圾内存并不是用完了马上就被释放,所以就会产生内存释放不及时的现象,从而降低了内存的使用。而当程序浩大的时候。这种现象更为明显,并且GC的工作也是需要消耗资源的。所以,也就会产生内存浪费。

JVM中的对象生命周期里谈内存回收:
      对象的生命周期一般分为7个阶段:创建阶段,应用阶段,不可视阶段,不可到达阶段,可收集阶段,终结阶段,释放阶段。
      创建阶段:首先大家看一下,如下两段代码:

      test1:
       for( int i=0; i<10000; i++)
            Object obj=new Object();

      test2:
      Object obj=null;
      for( int i=0; i<10000; i++)
            obj=new Object();

      这两段代码都是相同的功能,但是显然test2的性能要比test1性能要好,内存使用率要高,这是为什么呢?原因很简单,test1每次执行for循环都要创建一个Object的临时对象,但是这些临时对象由于JVM的GC不能马上销毁,所以他们还要存在很长时间,而test2则只是在内存中保存一份对象的引用,而不必创建大量新临时变量,从而降低了内存的使用。

      另外不要对同一个对象初始化多次。例如:

      public class A{
      private Hashtable table = new Hashtable();
      public A(){
      table = new Hashtable(); // 这里应该去掉,因为table已经被初始化.
      }
      }

      这样就new了两个Hashtable,但是却只使用了一个。另外一个则没有被引用.而被忽略掉.浪费了内存.并且由于进行了两次new操作.也影响了代码的执行速度。

      应用阶段:即该对象至少有一个引用在维护他.

      不可视阶段:即超出该变量的作用域。这里有一个很好的做法,因为JVM在GC的时候并不是马上进行回收,而是要判断对象是否被其他引用在维护.所以,这个时候如果我们在使用完一个对象以后对其obj=null或者obj.doSomething()操作,将其标记为空,可以帮助JVM及时发现这个垃圾对象.

      不可到达阶段:就是在JVM中找不到对该对象的直接或者间接的引用。

      可收集阶段,终结阶段,释放阶段:此为回收器发现该对象不可到达,finalize方法已经被执行,或者对象空间已被重用的时候。

JAVA的析构方法:
      可能不会有人相信,JAVA有析构函数? 是的,有。因为JAVA所有类都继承至Object类,而finalize就是Object类的一个方法,这个方法在JAVA中就是类似于C++析构函数.一般来说可以通过重载finalize方法的形式才释放类中对象.如:

      public class A{
      public Object a;
      public A(){ a = new Object ;}
      protected void finalize() throws java.lang.Throwable{
      a = null; // 标记为空,释放对象
      super.finalize();      // 递归调用超类中的finalize方法.
      }
      }

      当然,什么时候该方法被调用是由JVM来决定的.......................
      一般来说,我们需要创建一个destory的方法来显式的调用该方法.然后在finalize也对该方法进行调用,实现双保险的做法.
      由于对象的创建是递归式的,也就是先调用超级类的构造,然后依次向下递归调用构造函数,所以应该避免在类的构造函数中初始化变量,这样可以避免不必要的创建对象造成不必要的内存消耗.当然这里也就看出来接口的优势.

数组的创建:
      由于数组需要给定一个长度,所以在不确定数据数量的时候经常会创建过大,或过小的数组的现象.造成不必要的内存浪费,所以可以通过软引用的方式来告诉JVM及时回收该内存.(软引用,具体查资料).
      例如:

      Object obj = new char[10000000000000000];
      SoftReference ref = new SoftReference(obj);

共享静态存储空间:
      我们都知道静态变量在程序运行期间其内存是共享的,因此有时候为了节约内存工件,将一些变量声明为静态变量确实可以起到节约内存空间的作用.但是由于静态变量生命周期很长,不易被系统回收,所以使用静态变量要合理,不能盲目的使用.以免适得其反。
      因此建议在下面情况下使用:
      1,变量所包含的对象体积较大,占用内存过多.
      2,变量所包含对象生命周期较长.
      3,变量所包含数据稳定.
      4,该类的对象实例有对该变量所包含的对象的共享需求.(也就是说是否需要作为全局变量).

对象重用与GC:
      有的时候,如数据库操作对象,一般情况下我们都需要在各个不同模块间使用,所以这样的对象需要进行重用以提高性能.也有效的避免了反复创建对象引起的性能下降.
      一般来说对象池是一个不错的注意.如下:

      public abstarct class ObjectPool{
      private Hashtable locked,unlocked;
      private long expirationTime;
      abstract Object create();
      abstract void expire( Object o);
      abstract void validate( Object o);
      synchronized Object getObject(){...};
      synchronized void freeObject(Object o){...};
      }

      这样我们就完成了一个对象池,我们可以将通过对应的方法来存取删除所需对象.来维护这快内存提高内存重用.
      当然也可以通过调用System.gc()强制系统进行垃圾回收操作.当然这样的代价是需要消耗一些cpu资源.

不要提前创建对象:
      尽量在需要的时候创建对象,重复的分配,构造对象可能会因为垃圾回收做额外的工作降低性能.

JVM内存参数调优:
      强制内存回收对于系统自动的内存回收机制会产生负面影响,会加大系统自动回收的处理时间,所以应该尽量避免显式使用System.gc(),
      JVM的设置可以提高系统的性能.例如:
      java -XX:NewSize=128m -XX:MaxNewSize=128m -XX:SurvivorRatio=8 -Xms512m -Xmx512m
      具体可以查看java帮助文档.我们主要介绍程序设计方面的性能提高.

JAVA程序设计中有关内存管理的其他经验:
      根据JVM内存管理的工作原理,可以通过一些技巧和方式让JVM做GC处理时更加有效.,从而提高内存使用和缩短GC的执行时间.
      1,尽早释放无用对象的引用.即在不使用对象的引用后设置为空,可以加速GC的工作.(当然如果是返回值.....)
      2,尽量少用finalize函数,此函数是JAVA给程序员提供的一个释放对象或资源的机会,但是却会加大GC工作量.
      3,如果需要使用到图片,可以使用soft应用类型,它可以尽可能将图片读入内存而不引起OutOfMemory.
      4,注意集合数据类型的数据结构,往往数据结构越复杂,GC工作量更大,处理更复杂.
      5,尽量避免在默认构造器(构造函数)中创建,初始化大量的对象.
      6,尽量避免强制系统做垃圾回收.会增加系统做垃圾回收的最终时间降低性能.
      7,尽量避免显式申请数组,如果不得不申请数组的话,要尽量准确估算数组大小.
      8,如果在做远程方法调用.要尽量减少传递的对象大小.或者使用瞬间值避免不必要数据的传递.
      9,尽量在合适的情况下使用对象池来提高系统性能减少内存开销,当然,对象池不能过于庞大,会适得其反.

 

转自:米大老虎

分享到:
评论

相关推荐

    高性能java代码优化建议

    "高性能java代码优化建议"这一主题涵盖了众多资深开发者积累的实践经验,旨在帮助开发者编写出更高效、性能更强的代码。以下是一些关键的知识点: 1. **避免过度使用同步**:在多线程环境中,过多的同步可能导致...

    深入理解 Java 内存模型_程晓明_InfoQ_java_内存模型_

    理解JMM对于编写高效、正确且线程安全的Java代码至关重要。 Java内存模型规定了程序中的变量是如何在处理器之间以及内存系统中交互的。它包括了主内存(Main Memory)和每个线程的本地内存(Thread Local Memory)...

    高性能JAVA开发之内存管理

    ### 高性能JAVA开发之内存管理 在Java开发过程中,内存管理是确保程序高效稳定运行的关键环节之一。本文将深入探讨Java对象的生命周期,并通过具体的示例代码来解析不同阶段的特点及其对内存管理的影响。 #### ...

    SDMS.rar_java源代码_学生宿舍管理系统_学生宿舍管理系统 java_宿舍管理系统_寝室管理系统java源

    《学生宿舍管理系统JAVA源代码详解》 在信息技术领域,开发一套高效、易用的学生宿舍管理系统是提高学校后勤管理效率的重要手段。本系统基于JAVA编程语言,具有高度的可扩展性和稳定性,是学习和实践JAVA软件开发的...

    javamemory_JAVA内存监视器_java_

    Java内存监视是优化Java应用程序性能的关键环节,尤其是在处理大规模数据或长时间...通过合理地使用内存监视工具和API,我们可以有效地管理和优化内存使用,避免性能问题和内存异常,从而确保Java应用程序的高效运行。

    java-1.8.0_121.tar.gz

    它是Java 7及更高版本引入的,大大增强了开发者对JVM的控制和调试能力。 7. **安全性增强**: 1.8.0_121版本修复了多个安全漏洞,提高了Java应用程序的安全性。这包括修复了可能允许远程攻击者执行任意代码的漏洞,...

    Java内存管理机制相关资料汇总

    资源名称:Java内存管理机制相关资料汇总资源目录:【】java内存回收机制及预防【】java内存管理机制【】java内存...高性能Java代码之内存管理资源截图: 资源太大,传百度网盘了,链接在附件中,有需要的同学自取。

    commons-mmf.rar_java nio_java共享内存_共享内存

    Java NIO(New Input/Output)是Java标准库中提供的一种替代传统IO的高效I/O模型,它...通过合理利用`MappedByteBuffer`,开发者可以构建出高性能的多进程通信应用。然而,使用时必须注意同步问题,以防止数据不一致。

    Java虚拟机规范(Java_SE_7).pdf

    Java虚拟机(JVM)是Java程序运行的核心组件,它为Java代码提供了平台无关的运行环境。《Java虚拟机规范(Java_SE_7)》详细阐述了Java SE 7版本的JVM工作原理、内存模型以及指令集等关键概念。这份文档对于理解Java...

    java php python erlang 千万级内存数据性能比较

    在IT领域,尤其是在大数据处理和高性能计算中,选择正确的编程语言对于实现千万级内存数据的高效处理至关重要。本文将深入探讨Java、PHP、Python和Erlang这四种语言在处理千万级内存数据时的性能差异。 首先,让...

    java 1.8.0_45 for win10_64bit

    7. **Java Mission Control (JMC)** 和 **Java Flight Recorder (JFR)**:这两款工具在Java 8中开始提供,为高性能应用程序提供了高级诊断和监控功能。 8. **Java控制面板**:允许用户管理Java的设置,如更新、安全...

    engineMonitor_java.zip_Java服务器监控_Java监控_engineMonitor_engineMoni

    针对"engineMonitor_java.zip_Java服务器监控_Java监控_engineMonitor_engineMoni"这个压缩包文件,我们可以深入探讨Java服务器监控的相关知识点,以及它如何帮助我们管理和优化Java应用。 首先,"Java服务器监控...

    各种网络聊天源代码java_ c++ _p2p

    在聊天应用开发中,C++可以提供低级别的控制,允许开发者更精细地管理内存和系统资源。例如,使用Boost库中的Asio库可以简化网络编程,提供异步I/O操作,从而实现高效的聊天功能。 【标签】"聊天"、"java"、"cpp...

    Java练手_网吧管理系统.zip

    Java是一种高性能、跨平台的面向对象编程语言。它由Sun Microsystems(现在是Oracle Corporation)的James Gosling等人在1995年推出,被设计为一种简单、健壮、可移植、多线程、动态的语言。Java的主要特点和优势...

    JAVA代码优化工具

    Java代码优化是提升应用程序性能的关键步骤,尤其是在大型企业级应用或者高性能服务中。优化能够减少内存消耗,提高程序运行速度,降低CPU使用率,并改善整体的用户体验。在Java开发领域,有多种工具可以帮助开发者...

    从 Java 代码到 Java 堆 理解和优化您的应用程序的内存使用

    通过分析Java代码中的内存开销,以及讨论对象创建、集合效率等方面,将揭示如何确定性能瓶颈并改进代码。 【标签】:优化内存,Java内存管理,Java堆,对象创建,集合效率 【正文】: Java应用程序的内存管理是一个...

    java_se_6.zip

    Java SE 6,全称Java Standard Edition 6,是Oracle公司发布的Java开发平台的一个重要版本。这个版本在Java的历史上...通过深入学习和理解这个版本,开发者能够更好地掌握Java技术,编写出高质量、高性能的应用程序。

    java内存管理详细介绍.doc

    Java内存管理是Java编程中至关重要的一环,它与C++等其他...总的来说,虽然Java的内存管理自动化程度较高,但深入理解JVM的内存管理机制,能够帮助开发者编写更高效、更稳定的代码,避免不必要的性能瓶颈和内存问题。

    JNI_C_Java.zip_java 调c_jni_jni c_jni回调_编程之法

    JNI,全称Java Native Interface,是Java平台标准的一部分,它允许Java代码和其他语言写的代码进行交互。JNI在很多场景下都是必要的,比如当需要利用已有的C/C++库,或者需要提升性能时,我们通常会借助JNI来实现...

    基于GB32960实现车联网数据接入网关_java_代码_下载

    特点:高性能,高并发,高可用,支持K8S同POD多副本集群部署,横向拓展扩容 测试简介:基于4c8g云虚拟主机,IO密集型机器实测单节点TPS稳定高达13000/s。最长压测时间24小时。 更多详情、使用方法,请下载后阅读README.md...

Global site tag (gtag.js) - Google Analytics