欢迎光临
我们一直在努力

Java性能探寻-JSP教程,Java技巧及代码

建站超值云服务器,限时71元/月

在开发www.jobcn.com的期间,系统在运行的时候出现了一些问题,现将我的问题解决方法做一个小结,供以后作项目的参考。并请大家不吝指教.

问题1:

resin 的数据库连接数目在程序运行中不断增长,最后连接数目超过最大数,导致resin服务关闭

原因:

在操作完数据库后,没有关闭数据库连接;或者是返回结果集(resultset),而无法在jsp中关闭数据库连接。

解决方法:

1. 在操作完数据库要关闭数据库连接。

2. 尽量不要返回结果集resultset, 可以返回vector(一个字段)、hashtable(多个字段),这样可以在javabean中关闭数据库。

3. 如果javabean中返回的是结果集(resultset),也可以在javabean中写一个connectdb(连接数据库)、closedb(关闭数据库)的方法,然后jsp里面调用connectdb(),建立数据库连接,同时就可以对数据库进行操作了,操作数据库完毕,可以通过closedb() 来关闭数据库。

4.关闭连接时尽量使用conn.close(),rs.close()…..而不要另外写方法去关闭它,如调用db.closeconnection(conn)等等,这样其实对性能有一定影响,因为该性能经过了我对单个文件的测试.

5.关闭连接时请按顺序进行,如rs.close()..,stmt.close()..conn.close()….

问题2:

在运行某一个jsp 程序的时候,resin 的内存陡然增长,而且居高不下。最终导致resin 内存不足,甚至当机。

原因:

过度使用内存。

解决方法:

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 方法来进行字符串相连。

2. 在解决这个问题的时候,我也尝试使用上面的方法,效果并不是很明显(消耗内存上)。后来在显示大量数据的时候,避免字符串相连的步骤,而直接使用out.println(),直接输出。

问题3:java并不阻止程序占用过多的内存,当对象向堆所请求的内存不足时,垃圾收集器(garbage collector)就会自动启动,释放那些引用数为零的对象所占用的内存,java也不会自动释放无用的对象的引用,如果程序忘记释放指向对象的引用,则程序运行时的内存随着时间的推移而增加,发生所谓内存泄漏(memory leaks),创建对象不但消耗cpu的时间和内存,同时,为释放对象内存jvm需不停地启动垃圾收集器(garbage collector),这也会消耗大量的cpu时间。

解决方法: 由于在运行一段时间jsp程序后,resin 的内存会有一个缓慢的增长,这样也会导致内存溢出,为了避免这总现象的出现,最终的解决方法是: 编写一个servlet程序,在启动服务器的时候,启动一个这个servlet,每隔20分钟运行在服务器端运行一次,来定时回收内存。

问题4:

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 com.jobcn.company;

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

***** 说明:

***** 记录集:一条记录用放到一个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、把不用的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”,页面就会提交两次,(这个bug相信大家都知道其实没必要提出来)

在onclick事件里面会提交一次,而submit 按钮本身也会提交一次。

解决方法:

如果在onclick 事件里面提交页面,按钮的类型type 一定不可以是”submit或image” 按钮。

可以是”button”.

解决方法:

去掉这样<meta http-equiv="refresh" content="600">

的语句。

问题13:在跳转到别的页面的时候,要加return。

否则可能会引起错误。跳转不过去。

if{

request.getrequestdispatcher("/xxx/errorpage.jsp?s_mark=error:record have existed").forward(request,response);

return;

}

else

{

response.sendredirect();

//建议用上面的方法

return;

}

————————————————————————————————-

赞(0)
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com 特别注意:本站所有转载文章言论不代表本站观点! 本站所提供的图片等素材,版权归原作者所有,如需使用,请与原作者联系。未经允许不得转载:IDC资讯中心 » Java性能探寻-JSP教程,Java技巧及代码
分享到: 更多 (0)