JSP语言提高总结Word文档下载推荐.docx
《JSP语言提高总结Word文档下载推荐.docx》由会员分享,可在线阅读,更多相关《JSP语言提高总结Word文档下载推荐.docx(27页珍藏版)》请在冰豆网上搜索。
然而,这种技术在产品运行阶段对系统的资源是一个极大的损耗,因为它会给JSP引擎的类装载器(classloader)带来极大的负担。
因此关闭自动重载功能对系统性能的提升是一个极大的帮助。
1.3方法3:
不要滥用HttpSession
在很多应用中,我们的程序需要保持客户端的状态,以便页面之间可以相互联系。
但不幸的是由于HTTP具有天生无状态性,从而无法保存客户端的状态。
因此一般的应用服务器都提供了session来保存客户的状态。
在JSP应用服务器中,是通过HttpSession对像来实现session的功能的,但在方便的同时,它也给系统带来了不小的负担。
因为每当你获得或更新session时,系统者要对它进行费时的序列化操作。
你可以通过对HttpSession的以下几种处理方式来提升系统的性能。
如果没有必要,就应该关闭JSP页面中对HttpSession的缺省设置。
如果你没有明确指定的话,每个JSP页面都会缺省地创建一个HttpSession。
如果你的JSP中不需要使用session的话,那可以通过如下的JSP页面指示符来禁止它:
<%@pagesession="
false"
%>
不要在HttpSession中存放大的数据对像:
如果你在HttpSession中存放大的数据对像的话,每当对它进行读写时,应用服务器都将对其进行序列化,从而增加了系统的额外负担。
你在HttpSession中存放的数据对像越大,那系统的性能就下降得越快。
当你不需要HttpSession时,尽快地释放它:
当你不再需要session时,你可以通过调用HttpSession.invalidate()方法来释放它。
尽量将session的超时时间设得短一点:
在JSP应用服务器中,有一个缺省的session的超时时间。
当客户在这个时间之后没有进行任何操作的话,系统会将相关的session自动从内存中释放。
超时时间设得越大,系统的性能就会越低,因此最好的方法就是尽量使得它的值保持在一个较低的水平。
1.4方法4:
将页面输出进行压缩
压缩是解决数据冗余的一个好的方法,特别是在网络带宽不够发达的今天。
有的浏览器支持gzip(GNUzip)进行来对HTML文件进行压缩,这种方法可以戏剧性地减少HTML文件的下载时间。
因此,如果你将servlet或JSP页面生成的HTML页面进行压缩的话,那用户就会觉得页面浏览速度会非常快。
但不幸的是,不是所有的浏览器都支持gzip压缩,但你可以通过在你的程序中检查客户的浏览器是否支持它。
下面就是关于这种方法实现的一个代码片段:
publicvoiddoGet(HttpServletRequestrequest,HttpServletResponseresponse)
throwsIOException,ServletException{
OutputStreamout=null;
Stringencoding=request.getHeader("
Accept-Encoding"
if(encoding!
=null&
&
encoding.indexOf("
gzip"
)!
=-1){
request.setHeader("
Content-Encoding"
"
out=newGZIPOutputStream(request.getOutputStream());
elseif(encoding!
comdivss"
out=newZIPOutputStream(request.getOutputStream());
}else{
out=request.getOutputStream();
...
1.5方法5:
使用线程池
应用服务器缺省地为每个不同的客户端请求创建一个线程进行处理,并为它们分派service()方法,当service()方法调用完成后,与之相应的线程也随之撤消。
由于创建和撤消线程会耗费一定的系统资源,这种缺省模式降低了系统的性能。
但所幸的是我们可以通过创建一个线程池来改变这种状况。
另外,我们还要为这个线程池设置一个最小线程数和一个最大线程数。
在应用服务器启动时,它会创建数量等于最小线程数的一个线程池,当客户有请求时,相应地从池从取出一个线程来进行处理,当处理完成后,再将线程重新放入到池中。
如果池中的线程不够地话,系统会自动地增加池中线程的数量,但总量不能超过最大线程数。
通过使用线程池,当客户端请求急剧增加时,系统的负载就会呈现的平滑的上升曲线,从而提高的系统的可伸缩性。
1.6方法6:
选择正确的页面包含机制
在JSP中有两种方法可以用来包含另一个页面:
1、使用include指示符
<%@includeefile=”test.jsp”%>
2、使用jsp指示符
<jsp:
includeepage=”test.jsp”flush=”true”/>
在实际中发现,如果使用第一种方法的话,可以使得系统性能更高。
1.7方法7:
正确地确定javabean的生命周期
JSP的一个强大的地方就是对javabean的支持。
通过在JSP页面中使用jsp:
useBean标签,可以将javabean直接插入到一个JSP页面中。
它的使用方法如下:
useBeanid="
name"
scope="
page|request|session|application"
class="
package.className"
type="
typeName"
>
</jsp:
useBean>
其中scope属性指出了这个bean的生命周期。
缺省的生命周期为page。
如果你没有正确地选择bean的生命周期的话,它将影响系统的性能。
举例来说,如果你只想在一次请求中使用某个bean,但你却将这个bean的生命周期设置成了session,那当这次请求结束后,这个bean将仍然保留在内存中,除非session超时或用户关闭浏览器。
这样会耗费一定的内存,并无谓的增加了JVM垃圾收集器的工作量。
因此为bean设置正确的生命周期,并在bean的使命结束后尽快地清理它们,会使用系统性能有一个提高。
1.8其它一些有用的方法
1、在字符串连接操作中尽量不使用“+”操作符:
在java编程中,我们常常使用“+”操作符来将几个字符串连接起来,但你或许从来没有想到过它居然会对系统性能造成影响吧?
由于字符串是常量,因此JVM会产生一些临时的对像。
你使用的“+”越多,生成的临时对像就越多,这样也会给系统性能带来一些影响。
解决的方法是用StringBuffer对像来代替“+”操作符。
2、避免使用System.out.println()方法:
由于System.out.println()是一种同步调用,即在调用它时,磁盘I/O操作必须等待它的完成,因此我们要尽量避免对它的调用。
但我们在调试程序时它又是一个必不可少的方便工具,为了解决这个矛盾,我建议你最好使用Log4j工具(http:
//Jakarta.apache.org),它既可以方便调试,而不会产生System.out.println()这样的方法。
3、ServletOutputStream与PrintWriter的权衡:
使用PrintWriter可能会带来一些小的开销,因为它将所有的原始输出都转换为字符流来输出,因此如果使用它来作为页面输出的话,系统要负担一个转换过程。
而使用ServletOutputStream作为页面输出的话就不存在一个问题,但它是以二进制进行输出的。
因此在实际应用中要权衡两者的利弊。
1.9总结
本文的目的是通过对servlet和JSP的一些调优技术来极大地提高你的应用程序的性能,并因此提升整个J2EE应用的性能。
通过这些调优技术,你可以发现其实并不是某种技术平台(比如J2EE和.NET之争)决定了你的应用程序的性能,重要是你要对这种平台有一个较为深入的了解,这样你才能从根本上对自己的应用程序做一个优化。
2Jsp页面的位置
如果把这些JSP页面文件移到WEB-INF目录下,在调用页面的时候就必须把"
WEB-INF"
添加到URL中。
我们知道,实现页面的跳转有两种方式,一种是通过redirect的方式,一种是通过forward的方式。
redirect方式的跳转,系统会在一个新的页面打开要跳转的网页;
而forward方式跳转,系统会在原来的页面上打开一个要跳转的网页。
所以放到WEB-INF目录下的文件是不允许采用redirect方式的跳转来访问的,
如下例1:
/test/test1.jsp文件
<
html>
body>
formname="
testform"
action="
/WEB-INF/jsp/test/test.jsp"
>
inputtype="
submit"
value="
test"
/form>
/body>
/html>
上面这段语句只有一个名为test的按钮,如果单击这个按钮是,系统就会跳转到/WEB-INF/jsp/test/test.jsp,它的代码如下:
例2:
/WEB-INF/jsp/test/test.jsp文件
跳转成功!
事实上,这个跳转是无法成功的,点击按钮后,IE会报“403Forbidden”的错误。
而forward方式的跳转则可以成功,
如下代码:
例3:
/test/test2.jsp文件
jsp:
forwardpage="
/>
请注意上面红色的语句,这段就是通过forward的形式来访问/WEB-INF/jsp/test/test.jsp文件,在IE输入地址http:
//localhost/test1/test2.jsp,网页上就显示“跳转成功!
”的信息了,这表示放到了WEB-INF可以通过forward的方式来访问。
个人认为,像这种方式的可能不大时候采用一般jsp进行编程的系统,因为很多页面上都有采用submit这样的方式来进行跳转,但这种方式却非常适合采用struts结构的系统。
因为采用这个结果大多是先跳转到一个Action类,然后在Action类进行相关处理后(比如说获取相关的信息保存到session中,进行有效性的判断),然后再forward到另外一个页面,这样放到WEB-INF中的jsp代码可以被正常访问,也防止了对这些页面的直接访问,下面我来举例说明。
下面我们先对配置文件struts-config.xml进行配置,如下:
例4:
WEB-INF/struts-config.xml文件
?
xmlversion="
1.0"
encoding="
ISO-8859-1"
?
!
DOCTYPEstruts-configPUBLIC"
-//ApacheSoftwareFoundation//DTDStrutsConfiguration1.1//EN"
"
http:
//jakarta.apache.org/struts/dtds/struts-config_1_1.dtd"
struts-config>
--==========ActionMappingDefinitions===================-->
action-mappings>
action
path="
/test"
type="
test.TestAction"
scope="
request"
forwardname="
path="
/>
/action>
/action-mappings>
/struts-config>
上面这个配置非常简单,这里定义了一个action类,它的路径为/test,所对应的类为test.TestAction.java,它都一个跳转页面,别名为test,对应的页面为/WEB-INF/jsp/test/test.jsp。
下面我们对例1的内容进行修改,使其跳转到/test去。
例5:
修改后的/test/test1.jsp文件
<
这样我们在IE中访问http:
//localhost/test/test1.jsp,然后点击test按钮,页面就会跳转到test.TestAction.java这个类来,下面是这个类的内容。
packagetest;
importjavax.servlet.http.*;
importorg.apache.struts.action.ActionMapping;
importorg.apache.struts.action.Action;
importorg.apache.struts.action.ActionForm;
importorg.apache.struts.action.ActionForward;
publicclassTestActionextendsAction
{
publicActionForwardperform(ActionMappingmapping,ActionFormform,HttpServletRequestreq,HttpServletResponseres)
{returnmapping.findForward("
}
可以看到,这个类是继承Action类的,所有的控制类都必须继承Action类,这个类里面有一个perform方法,跳转到这个类都是从这个方法进行访问的(新版本可以是execute方法),现在这个方法里面只有一条语句,这句话的意思就是跳转到一个别名为test的页面,也就是/WEB-INF/jsp/test/test.jsp页面,这样我们点击test按钮后,IE就会显示“跳转成功!
”这条信息,这表示系统允许这样的跳转。
注意:
CSS文件要存放在根目录,此时引用要用绝对路径!
!
引用格式如:
/项目名/css文件夹/*.css标准的MVC模型建议把JSP放在WEB-INF下!
3JSP页面结构
3.1一:
JSP注释
3.1.1可以输出的注释(HTML注释)
语法:
--comment[<
%=expression%>
]-->
输出的注释是发送到浏览器端的,可以在网页的源码中看到它。
3.1.2隐藏的注释(JSP标准注释)
%--comment--%>
用隐藏的注释标记的字符在JSP编译时被忽略,也不会显示在客户浏览器中。
3.1.3Java语言注释
%/*comment*/%>
标记的字符在JSP编译时被忽略,不会显示在客户浏览器中。
%@pagelanguage="
java"
contentType="
text/html;
charset=gb2312"
import="
java.sql.*"
errorPage="
"
%>
%="
我是杨天赐,这是我的JSP网站!
%--这是JSP页面--%>
3.2二:
JSP指令(编译指令)
3.2.1page指令
该指令用來定义当前页面的属性。
%@page{屬性="
屬性值"
}%>
屬性:
∙language="
∙contentType="
charset=GBK"
∙import="
java.util.*,java.sql.*"
java.lang.*,java.servlet.*,java.servlet.jsp.*,java.servlet.http.*
∙session="
true|false"
∙buffer="
none|8kb|sizekb"
∙autoFlush="
∙errorPage="
url"
∙isErrorPage="
3.2.2include指令
用來向当前页面中插入一个静态文件的內容。
%@includefile="
filename"
%>
3.2.3taglib指令
用来定义一个标签库及其自定义标签的前缀。
%@tagliburl="
tagLibraryURL"
prefix="
tagPrefix"
--用來向当前页面中插入一个静态文件的內容。
-->
hello.jsp"
3.3三:
JSP动作(操作指令)
3.3.1jsp:
include指令
includepage="
relativeURL"
flush="
true"
用于在当前JSP页面中加入静态或动态的文件资源。
3.3.2jsp:
forward指令
forwardpage="
用于引导客户端请求到另一个页面或另一个Servlet中。
3.3.3jsp:
param指令
paramname="
parameterName"
parameterValue"
向一个动态文件发送一个或多个参数。
--用于在当前JSP页面中加入静态或动态的文件资源。
--服务器重定向-->
param_forward:
param_request.jsp"
schl"
sje"
addrs"
hefei"
/jsp:
forward>
param_request:
%
request.setCharacterEncoding("
GBK"
Stringstrschl="
;
Stringstraddrs="
strschl=request.getParameter("
straddrs=request.getParameter("
学校:
%=strschl%>
br>
地址:
%=straddrs%>
3.4四:
JSP脚本元素
3.4.1JSP声明
%!
声明一;
声明二;
…%>
用来定义一个或多个合法的变量(普通变量和类变量)和方法,并在JSP页面初始化时被初始化。
∙声明必须以“;
”结尾
∙一个声明仅在一个页面中有效
∙被page指令包含进来的已经声明的变量和方法不需重新声明
3.4.2JSP表达式
%=expression%>
3.4.3JSP脚本小程序(Scriptlet)
%Scriptlet的代码段%>
一个脚本小程序可以包含多个JSP语句、方法、变量、表达式等。
∙在代码块中声明的变量是JSP页面的局部变量,只在当前页面中有效。
%@pagecontentType="
language="
%
inti=0;
i++;
out.print(i);
你是第<
%=i%>
个访问该网站的人!
inti=0;