最近在JSP程序开发过程中,发现了很严重的Java内存溢出问题。我采用了Resin服务器自动重启的方法,临时保证了网站的正常运行。但是我想去寻找问题的症结之所在,我想一定是JSP程序或Java程序没有及时地释放内存,但到底是哪些类的哪些对象没有被释放?   我认真的看了一下Resin的错误日志(error.log文件),发现程序出现java.lang.OutOfMemoryError的错误大多情况是在运行一个上传附件的文件时发生的。在这个上传附件的JSP文件中,我调用了目前网上很流行的jspsmartupload组件,难道是这个组件本身有问题?   于是我就下载了一个Java反编译工具,把jspsmartupload组件的源码文件(class文件)反编译成Java文件。这个组件用到了五个自定义的类,分别是File类、Files类、Request类、SmartUpload类和SmartUploadException类。里面最核心的方法莫过于SmartUpload类的upload()方法。我看到upload方法中有这样两个语句:      m_totalBytes = m_request.getContentLength();     (*)         m_binArray = new byte[m_totalBytes];                    (**)     其中,变量m_totalBytes表示用户上传的文件(二进制文件)的内容的总长度,我做了一个测试,一个55.3M 的文件,它的m_totalBytes是58012662 ,显然这是一个很大的数。如果用这样大的数去声明一个byte数组,并给数组的每个元素分配内存空间,Java内存不溢出才怪呢!另外,我发现SmartUpload类中的m_binArray数组竟然从来没有被释放,难道真的不需要释放吗?JVM的功能有我们想象的那么强大吗?    我对SmartUpload类作了如下的改进:    (1)对upload()方法的(**)语句加上了totalBytes大小的限制,并加上了try...catch...finally去处理异常,如下:        try{         if(m_totalBytes>55184652 || m_totalBytes<0)    //55184652相当于52.6M         {          throw new SmartUploadException("the file is too large to upload.");         }         else         {                           m_binArray = new byte[m_totalBytes];                           ......                }             catch(Exception e){......}             catch(Error err){......}             finally{  m_binArray=null;}       这样,就能保证不管用户传的文件是多大,不会出现java.lang.OutOfMemoryError的错误,而且如果upload失败,m_binArray所占的内存空间也会被释放。     (2)在SmartUpload类中增加了一个stop()方法,该方法完成的工作就是释放m_binArray数组所占的内存空间。代码如下:     try{      m_binArray=null;       }     catch(Exception e)     {      System.out.println(e.toString());       }     jspsmartupload组件就做了这些修改,下面就需要更改调用该组件的JSP文件了。其实这个JSP文件有一些地方写得不规范。(以下代码中mySmartUpload表示SmartUpload的一个实例化对象)     (1)首先要设置上传附件的MaxSize。                    mySmartUpload.setTotalMaxFileSize(10240000); //10240000表示10M     (2)调用SmartUpload.upload()方法要捕获一个异常。              try{            mySmartUpload.upload();                    }                    catch(Exception e)                   {               out.print("文件太大,无法上传");               return;                   }     (3)程序结束时要调用SmartUpload.stop()方法。                  mySmartUpload.stop();                            mySmartUpload=null;     好了,修改完了,把修改后的文件用Resin进行测试(操作系统仍然用Linux9.0),程序至今还没有出现java.lang.OutOfMemoryError的错误。    但愿这种方法能彻底帮我解决问题。当然,还需做进一步测试,看看修改后的组件会不会出现新问题。                        

03-15 04:39