weblogic記憶體佔用等問題的解決方法

quanshengaa發表於2015-11-06

在做專案的時候,jsp在執行的時候出現了一些問題,現將我的問題解決方法做一個小結,供以後作專案的參考。

問題1:
weblogic 的資料庫連線數目在程式執行中不斷增長,最後連線數目超過最大數,導致weblogic服務關閉
<!-- Inject Script Filtered --> <!-- Inject Script Filtered -->
原因:
在操作完資料庫後,沒有關閉資料庫連線;或者是返回結果集(Resultset),而無法在jsp中關閉資料庫連線。
解決方法:
1. 在操作完資料庫要關閉資料庫連線。
2. 儘量不要返回結果集Resultset, 可以返回Vector(一個欄位)、Hashtable(多個欄位),這樣可以在javabean中關閉資料庫。
3. 如果javabean中返回的是結果集(Resultset),也可以在javaBean中寫一個connectDB(連線資料庫)、closeDB(關 閉資料庫)的方法,然後jsp裡面呼叫connectDB(),建立資料庫連線,同時就可以對資料庫進行操作了,運算元據庫完畢,可以透過closeDB () 來關閉資料庫。

問題2:
在執行某一個jsp 程式的時候,weblogic記憶體陡然增長,而且居高不下。最終導致weblogic 記憶體不足,甚至當機。
原因:
過度使用記憶體
解決方法:
1. 由於資料量比較大,在對字串進行操作的時候,使用 + 進行字串連線,而
相信大家對String都非常熟悉,我們也經常要用它來做字串的連線什麼的,例如:
String a =b+c file://b,c 都是String
但是在實際的編譯中卻是這樣:
String a=new StringBuffer().append(b).append(c).toString() 顯然,在一個簡單的語句中卻意外的多生成了2個物件:
.StringBuffer()
.toString返回的一個String
我們比較一下這兩段程式的效能:
程式片斷一:
StringBuffer s=new StringBuffer();
long start = System.currentTimeMillis();
for (int i=0;i<10000;i++){
s1+="a";
}
long stop = System.currentTimeMillis();
System.out.println(stop-start);
程式片斷二:
StringBuffer s=new StringBuffer(10000);//
long start=System.currentTimeMillis();
for (int i=0;i<10000;i++){
s.append("a");
}
long stop=System.currentTimeMillis();
System.out.println(stop-start);
比較一下結果,差距很明顯。
至於為什麼String的連線這麼做,因為String無法直接改變其長度,而必須採用StringBuffer的用法。
因此建議使用StringBuffer 的append 方法來進行字串相連。

[@more@]2. 在解決這個問題的時候,我也嘗試使用上面的方法,效果並不是很明顯(消耗記憶體上)。後來在顯示大量資料的時候,避免字串相連的步驟,而直接使用out.println(),直接輸出。
問題3:java並不阻止程式佔用過多的記憶體,當物件向堆所請求的記憶體不足時,垃圾收集器(Garbage Collector)就會自動啟動,釋放那些引用數為零的物件所佔用的記憶體,java也不會自動釋放無用的物件的引用,如果程式忘記釋放指向物件的引用,則程式執行時的記憶體隨著時間的推移而增加,發生所謂記憶體洩漏(memory leaks),建立物件不但消耗CPU的時間和記憶體,同時,為釋放物件記憶體JVM需不停地啟動垃圾收集器(Garbage Collector),這也會消耗大量的CPU時間。

解決方法: 由於在執行一段時間jsp程式後,weblogic記憶體會有一個緩慢的增長,這樣也會導致記憶體溢位,為了避免這總現象的出現,最終的解決方法是: 編寫一個servlet程式,在啟動伺服器的時候,啟動一個這個servlet,每隔20分鐘執行在伺服器端執行一次,來定時回收記憶體
問題4:
log檔案裡面的除錯資訊沒有註釋去掉。
解決方法:
在程式透過後,儘量把除錯的資訊註釋去掉。
同時在捕捉錯誤的時候要寫明程式名稱,方便查詢,這一點做的還不夠。
最好能寫一個記錄log的方法,以便程式呼叫。
問題5
改善效能,提高速度。
具體例項:
我們再來看一個有關Vector類的程式碼片段:
for(int I=0; I<v.size(); I++) { System.out.println( v.get(I).getClass().toString()); }
如果v包含100,000個元素,這個程式碼片段將呼叫v.size()方法100,000次。雖然size方法是一個簡單的方法,但它仍舊需要一次方法調 用的開銷,至少JVM需要為它配置以及清除堆疊環境。在這裡,for迴圈內部的程式碼不會以任何方式修改Vector型別物件v的大小,因此上面的程式碼最好 改寫成下面這種形式:
int size = v.size(); for(int I=0; I<size; I++) { System.out.println( v.get(I).getClass().toString()); }
雖然這是一個簡單的改動,但它仍舊贏得了效能。畢竟,每一個CPU週期都是寶貴的。
問題 6:
在 jsp 檔案裡面 不要寫 <%@ page import="java.lang.*" %>
因為java 不需要引入此包就可以引用裡面的類檔案。

問題7:
使用vector+hashtable 一次返回查詢結果resulset.

解決方法: 記錄集:將一條記錄放到一個hashtable裡面,然後把它再
新增到vector裡面,迴圈記錄下結果集,返回vector
具體見後面的java檔案的部分程式碼(不包擴資料庫的連線和關閉)


package zjdx.bean.common;

/********************************************
***** Title: hashtable_vector_rs
***** Description: 資料顯示
***** Copyright: Copyright (c) 2002
***** Company: DHC
***** author: wangyl
***** version: 1.0
***** 說明:
***** 記錄集:一條記錄用放到一個hashtable裡面,然後把它再
放到vector裡面,迴圈記錄下結果集,返回vector

*********************************************/

import java.io.*;
import java.sql.*;
import java.util.*;
import java.text.*;

public class hashtable_vector_rs
{
/*----------------------------------------------------------------*/
/* 函式名稱: getMultiRowInfo
/* 功能描述: 返回記錄集,放到Hashtable裡面
/* 引數: sql 語句,欄位個數
/* 返回值: 成功---htable, 失敗---null
/*----------------------------------------------------------------*/
public Vector ListResult(String sqlStatement,int num)
{
Vector ListRs=new Vector();

try
{
connectDB();
rs=stmt.executeQuery(sqlStatement);
//判斷欄位資料型別
//date型 返回 93
//int型 返回 2,4
//bigint型 返回 3
//String型 返回 12
//(char 型)返回 1
int t = -1;
rsmd = rs.getMetaData();
int ColumnCount=0;
if(num>0)
ColumnCount = num;
else
ColumnCount = rsmd.getColumnCount();
while(rs.next())
{
Hashtable htable =new Hashtable();
for (int i = 1; i <= ColumnCount; i++)
{
t=rsmd.getColumnType(i);
System.out.println("i="+i+",t="+t+"name="+rsmd.getColumnName(i));
if(t==12||t==1||t==3)
{
if(rs.getString(i)==null|| rs.getString(i).equals(""))
htable.put(rsmd.getColumnName(i),"");
else
htable.put(rsmd.getColumnName(i),rs.getString(i));
}
else if(t==93)
{
htable.put(rsmd.getColumnName(i),rs.getDate(i).toString());
}
else if(t==2||t==4)
{
htable.put(rsmd.getColumnName(i),Integer.toString(rs.getInt(i)));
}
}
ListRs.add(htable);
}//e
return ListRs;
}
catch(Exception listError)
{
System.out.println("資料庫操作失敗!"+listError);
return null;
}
finally
{
try
{
closeDB();
}
catch(Exception closeErr)
{
System.out.println("關閉資料庫出錯:"+closeErr);
}
}
}
}

問題8: jsp的程式也有設計的不夠合理的地方
例如:選擇一個下拉框,提交一次,列出所選的資料,選擇另外一個下拉框再次提交,再次列出所選的資料。
解決方法:
儘量一次把條件選擇完畢,然後列出所選擇的資料,並且在資料多的時候,儘量使用翻頁,減少執行時間。

問題9:效能最佳化,儘量使用PreparedStatement
解決方法:
PreparedStatement 物件和使用的普通的 Statement 物件有兩點不同。
第一, 它們是為了效能更快而由 JDBC 驅動程式或資料庫編譯(預編譯)的。
第二, 它們接受一個或多個動態輸入引數,稱為 IN 引數。這兩點讓 PreparedStatement 物件適用於重複的 SQL 操作,其中操作基本上都是一樣的,只有微小的差異(如資料載入)。要讓 SQL 語句在使用前預備好,在 PreparedStatement 物件建立時必須將 SQL 傳送到 JDBC 驅動程式,而不是在其執行時才傳送。

IN 引數用 SQL String 中的 ? 佔位符表示。在 PreparedStatement 能夠成功地執行前,還必須呼叫 PreparedStatement 物件的 setXXX() 方法來設定 IN 引數,在這裡 XXX 被設定的引數的資料型別所替換。因而,要將第一個 IN 引數設定為整數值 100,您應該呼叫 setInt(1, 100)。同樣地,要將第二個 IN 引數設定為字串值“rjb”,您應該呼叫 setString(2, "rjb")。最後一點是,設定好的引數值在設定為一個新的值,或用 clearParameters() 顯式地清除之前會保持不變。這很重要,因為 PreparedStatement 可以被多次執行;如果您不注意的話,就會讓您的資料庫充滿無用資料。
問題10 :
1、我把包含SQLBridge 的java檔案改了一遍,有32個java檔案(已經改完)。
2、我把包含SQLPool 的java檔案改了一遍,有27個java檔案(已經改完)。
3、我把不用的jsp檔案整理了一下,大概有17 個目錄的jsp檔案不用或者沒有上線。(整理完畢)
問題11:提交頁面,每次都提交兩次。
原因:
1、
<input type="submit" name="sendit" value="傳送" class=button onclick="javascript:if(chkit())form_submit('broadcast_operation.jsp?s_coming=1&s_num=<%=str_num%>')">
這裡如果type=”sbumit”,頁面就會提交兩次。(花費了一天的時間才找到原因)。
在onclick事件裡面會提交一次,而submit 按鈕本身也會提交一次。

解決方法:
如果在onclick 事件裡面提交頁面,按鈕的型別type 一定不可以是”submit” 按鈕。
可以是”button”.
全文搜尋共有33個類似的檔案。
java.net.SocketException: ReadFile failed: 指定的網路名不再可用。
主要是由於這個引起的。
問題12:定時重新整理頁面,600秒(不是必要的,不要這樣做)
原因:
<meta http-equiv="refresh" content="600">
解決方法:
去掉這樣的語句。

全文搜尋共有12個類似的檔案。

問題13:在跳轉到別的頁面的時候,要加return。
否則可能會引起錯誤。跳轉不過去。
If{
request.getRequestDispatcher("/zjdx/jsp/common/ErrorPage.jsp?s_mark=error:record have existed").forward(request,response);
return;
}
else
{
response.sendRedirect();
//建議用上面的方法
return;
}

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/15797451/viewspace-1824833/,如需轉載,請註明出處,否則將追究法律責任。

相關文章