JSP基础

Java Server Pages

一种跨平台的动态网页技术标准,是在传统的网页HTML文件中插入Java程序段(Scriptlet)和JSP标记(tag),后缀名为(*.JSP)。

JSP是一种Java servlet,主要用于实现Java web应用程序的用户界面部分。
网页开发者们通过结合HTML代码、XHTML代码、XML元素以及嵌入JSP操作和命令来编写JSP。
通过网页表单获取用户输入数据、访问数据库及其他数据源,然后动态地创建网页。

JSP的优势
  • 与CGI相比:
  1. 性能更加优越,因为JSP可以直接在HTML网页中动态嵌入元素而不需要单独引用CGI文件。
  2. 服务器调用的是已经编译好的JSP文件,而不像CGI/Perl那样必须先载入解释器和目标脚本。
  3. JSP基于Java Servlets API,因此,JSP拥有各种强大的企业级Java API,包括JDBC,JNDI,EJB,JAXP等等。
  4. JSP页面可以与处理业务逻辑的servlets一起使用,这种模式被Java servlet 模板引擎所支持。
  • 与ASP相比:JSP有两大优势。首先,动态部分用Java编写,而不是VB或其他MS专用语言,所以更加强大与易用。第二点就是JSP易于移植到非MS平台上。
  • 与纯 Servlets相比:JSP可以很方便的编写或者修改HTML网页而不用去面对大量的println语句。
  • 与SSI相比:SSI无法使用表单数据、无法进行数据库链接。
  • 与JavaScript相比:虽然JavaScript可以在客户端动态生成HTML,但是很难与服务器交互,因此不能提供复杂的服务,比如访问数据库和图像处理等等。
  • 与静态HTML相比:静态HTML不包含动态信息。
Tomcat

Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,属于轻量级应用服务器,适合开发和调试JSP 程序。

Tomcat是Apache 服务器的扩展,独立运行。
当配置正确时,Apache 为HTML页面服务,而Tomcat 实际上运行JSP 页面和Servlet。另外,Tomcat和IIS等Web服务器一样,具有处理HTML页面的功能,另外它还是一个Servlet和JSP容器,独立的Servlet容器是Tomcat的默认模式。不过,Tomcat处理静态HTML的能力不如Apache服务器。

  • 安装和使用
  1. http://tomcat.apache.org/ 上下载最新版本的 Tomcat。
  2. 解压缩到一个方便的位置,通过/bin/startup.bat启动Tomcat。
  3. Tomcat 启动后,可以通过在浏览器地址栏输入 http://localhost:8080/ 访问 Tomcat 中的默认应用程序。
  • 目录结构
目录说明
/bin存放windows或Linux平台上启动和关闭Tomcat的脚本文件
/conf存放Tomcat服务器的各种全局配置文件,其中最重要的是server.xml和web.xml
/webappsTomcat的主要Web发布目录,默认情况下把Web应用文件放于此目录,存放Tomcat自带的两个WEB应用admin应用和 manager应用
/work存放JSP编译后产生的class文件
/lib存放Tomcat服务器所需的各种JAR文件
/common/lib存放Tomcat服务器以及所有web应用都可以访问的jar文件
/shared/lib存放所有web应用都可以访问的jar文件(但是不能被Tomcat服务器访问)
/doc存放Tomcat文档
/temp存放临时文件
/logs存放Tomcat执行时的日志文件
/src存放Tomcat的源代码
JSP处理过程
  • 就像其他普通的网页一样,您的浏览器发送一个HTTP请求给服务器。
  • Web服务器识别出这是一个对JSP网页的请求,并且将该请求传递给JSP引擎。通过使用URL或者.JSP文件来完成。
  • JSP引擎从磁盘中载入JSP文件,然后将它们转化为servlet。这种转化只是简单地将所有模板文本改用println()语句,并且将所有的JSP元素转化成Java代码。
  • JSP引擎将servlet编译成可执行类,并且将原始请求传递给servlet引擎。
  • Web服务器的某组件将会调用servlet引擎,然后载入并执行servlet类。在执行过程中,servlet产生HTML格式的输出并将其内嵌于HTTP response中上交给Web服务器。
  • Web服务器以静态HTML网页的形式将HTTP response返回到您的浏览器中。
  • 最终,Web浏览器处理HTTP response中动态产生的HTML网页,就好像在处理静态网页一样。

在这里插入图片描述

JSP生命周期
  • 编译阶段:
    servlet容器编译servlet源文件,生成servlet类。当浏览器请求JSP页面时,JSP引擎会首先去检查是否需要编译这个文件。如果这个文件没有被编译过,或者在上次编译后被更改过,则编译这个JSP文件。
  • 初始化阶段:
    加载与JSP对应的servlet类,创建其实例,并调用它的初始化方法JSPInit(),可自定义重写初始化方法。
  • 执行阶段:
    调用与JSP对应的servlet实例的服务方法。当JSP网页完成初始化后,JSP引擎将会调用_JSPService()方法
    _JSPService()方法需要一个HttpServletRequest对象和一个HttpServletResponse对象作为它的参数。
    _JSPService()方法在每个request中被调用一次并且负责产生与之相对应的response,并且它还负责产生所有7个HTTP方法的回应
  • 销毁阶段:
    调用与JSP对应的servlet实例的销毁方法,然后销毁servlet实例。JSPDestroy()方法在JSP中等价于servlet中的销毁方法。当您需要执行任何清理工作时复写JSPDestroy()方法,比如释放数据库连接或者关闭文件夹等等。
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<html>
<head>
<title>life.JSP</title>
</head>
<body>
<%!
  public void JSPInit(){
  //初始化
  }
  public void JSPDestroy(){
  //销毁
  }
%>
<%
void _JSPService(HttpServletRequest request,HttpServletResponse response)
{
   // 服务端处理代码
}
%>
</body>
</html>
JSP语法
  • 脚本程序Scriptlet

脚本程序可以包含任意量的Java语句、变量、方法或表达式,只要它们在脚本语言中是有效的。不能定义静态变量或者常量。

<% 代码片段 %>

  • JSP声明

声明语句可以声明全局变量、方法,供后面的Java代码使用。在JSP文件中,您必须先声明这些变量和方法然后才能使用它们。可以定义静态变量或者常量。

<%! declaration; [ declaration; ]+ ... %>

  • JSP表达式

一个JSP表达式中包含的脚本语言表达式,先被转化成String,然后插入到表达式出现的地方。
由于表达式的值会被转化成String,所以您可以在一个文本行中使用表达式而不用去管它是否是HTML标签。

<%= 表达式 %>
<p>   
  Today's date: <%= (new java.util.Date()).toLocaleString()%>
</p>
  • JSP注释
语法描述
< %-- 注释 --%>JSP注释,注释内容不会被发送至浏览器甚至不会被编译
< ! – 注释 -->HTML注释,通过浏览器查看网页源代码时可以看见注释内容
  • 控制语句
<% if (day == 1 | day == 7) { %>      
  <p> Today is weekend</p>
<% } else { %>      
  <p> Today is not weekend</p>
<% } %>
<%  
switch(day) { 
case 0:    
  out.println("It\'s Sunday.");    
  break; 
case 1:    
  out.println("It\'s Monday.");    
  break; 
case 2:    
  out.println("It\'s Tuesday.");    
  break; 
case 3:    
  out.println("It\'s Wednesday.");    
  break; 
case 4:    
  out.println("It\'s Thursday.");    
  break; 
case 5:    
  out.println("It\'s Friday.");    
  break; 
default:    
  out.println("It's Saturday."); 
} 
%>
<%for ( fontSize = 1; fontSize <= 3; fontSize++){ %>   
  <font color="green" size="<%= fontSize %>">    
  JSP Tutorial   
  </font><br />
<%}%>
  • 常量
    布尔值(boolean):true 和 false;
    整型(int):与Java中的一样;
    浮点型(float):与Java中的一样;
    字符串(string):以单引号或双引号开始和结束;
    Null:null。
  • 运算符
    JSP支持所有Java逻辑和算术运算符。
JSP指令
<%@ directive attribute="value" %>
指令描述
<%@ page … %>定义页面的依赖属性,比如脚本语言、error页面、缓存需求等等
<%@ include … %>包含其他文件
<%@ taglib … %>引入标签库的定义,可以是自定义标签

包含文件的方式有两种:
<%@ include file=”…”%> 静态包含,先包含,后编译处理
< JSP:include page=”…”> 动态包含,先编译处理,后包含

  • Page指令

为容器提供当前页面的使用说明。

Page指令的语法格式:

<%@ page attribute="value" %>

等价的XML格式:

<JSP:directive.page attribute="value" />
属性描述
buffer指定out对象使用缓冲区的大小
autoFlush控制out对象的 缓存区
contentType指定当前JSP页面的MIME类型和字符编码
errorPage指定当JSP页面发生异常时需要转向的错误处理页面
isErrorPage指定当前页面是否可以作为另一个JSP页面的错误处理页面
extends指定servlet从哪一个类继承
import导入要使用的Java类
info定义JSP页面的描述信息
isThreadSafe指定对JSP页面的访问是否为线程安全
language定义JSP页面所用的脚本语言,默认是Java
session指定JSP页面是否使用session
isELIgnored指定是否执行EL表达式
isScriptingEnabled确定脚本元素能否被使用
  • Include指令

包含其他文件,如JSP文件、HTML文件或文本文件,成为该JSP文件的一部分,被同时编译执行。

Include指令的语法格式如下:

<%@ include file="relative url" %>

Include指令中的文件名实际上是一个相对的URL。如果您没有给文件关联一个路径,JSP编译器默认在当前路径下寻找。
等价的XML语法:

<JSP:directive.include file="relative url" />
  • Taglib指令

JSP API允许用户自定义标签,一个自定义标签库就是自定义标签的集合。
Taglib指令引入一个自定义标签集合的定义,包括库路径、自定义标签。

Taglib指令的语法:

<%@ taglib uri="uri" prefix="prefixOfTag" %>

uri属性确定标签库的位置,prefix属性指定标签库的前缀。
等价的XML语法:

<JSP:directive.taglib uri="uri" prefix="prefixOfTag" />
JSP行为(动作)

JSP行为标签使用XML语法结构来控制servlet引擎。它能够动态插入一个文件,重用JavaBean组件,引导用户去另一个页面,为Java插件产生相关的HTML等等。
与JSP指令元素不同的是,JSP动作元素在请求处理阶段起作用。

动作元素只有一种语法,它符合XML标准:

<JSP:action_name attribute="value" />
语法描述
JSP:include用于在当前页面中包含静态或动态资源
JSP:useBean寻找和初始化一个JavaBean组件
JSP:setProperty设置 JavaBean组件的值
JSP:getProperty将 JavaBean组件的值插入到 output中
JSP:forward从一个JSP文件向另一个文件传递一个包含用户请求的request对象
JSP:plugin用于在生成的HTML页面中包含Applet和JavaBean对象
JSP:element动态创建一个XML元素
JSP:attribute定义动态创建的XML元素的属性
JSP:body定义动态创建的XML元素的主体
JSP:text用于封装模板数据
  • 属性

id:动作元素的标识,在JSP页面中引用。通过PageContext来调用

scope:识别动作元素的生命周期,有四个值:page,requeet,session,application。

JSP九大隐含对象
对象描述
requestHttpServletRequest类的实例
responseHttpServletResponse类的实例
outPrintWriter类的实例,用于把结果输出至网页上
sessionHttpSession类的实例
applicationServletContext类的实例,与应用上下文有关
configServletConfig类的实例
pageContextPageContext类的实例,提供对JSP页面所有对象以及命名空间的访问
page类似于Java类中的this关键字
ExceptionException类的对象,代表发生错误的JSP页面中对应的异常对象
  • request对象

当客户端请求一个JSP页面时,JSP引擎就会制造一个新的request对象来代表这个请求

request对象提供了一系列方法来获取HTTP头信息,cookies,HTTP方法等等。

  • response对象

服务器创建request对象时会同时创建用于响应这个客户端的response对象。

response对象也定义了处理HTTP头模块的接口。通过这个对象,开发者们可以添加新的cookies,时间戳,HTTP状态码等等。

  • out对象

用来在response对象中写入内容,即向客户端输出数据,同时还可以管理应用服务器上的缓存。常用的方法有:out.print()、out.println()、out.flush()。

JSPWriter类包含了大部分java.io.PrintWriter类中的方法,并且新增了一些专为处理缓存而设计的方法。JSPWriter类会抛出IOExceptions异常,而PrintWriter不会。

  • session对象

用来跟踪在各个客户端请求间的会话

  • application对象

这个对象在JSP页面的整个生命周期中都代表着这个JSP页面。这个对象在JSP页面初始化时被创建,随着JSPDestroy()方法的调用而被移除。

通过向application中添加属性,则所有组成您web应用的JSP文件都能访问到这些属性

  • config对象

这个对象允许开发者访问Servlet或者JSP引擎的初始化参数,可以是属性名和属性值的参数,也可以是通过ServletContext对象传递的服务器相关信息,比如文件路径等。

  • pageContext对象

这个对象主要用来访问页面信息,同时过滤掉大部分实现细节。

这个对象存储了request对象和response对象的引用。application对象,config对象,session对象,out对象可以通过访问这个对象的属性来导出。

pageContext对象也包含了传给JSP页面的指令信息,包括缓存信息,ErrorPage URL,页面scope等。

PageContext类定义了一些字段,包括PAGE_SCOPE,REQUEST_SCOPE,SESSION_SCOPE, APPLICATION_SCOPE。它也提供了40余种方法,有一半继承自javax.servlet.JSP.JSPContext 类。

pageContext.getRequeset();
pageContext.getSession();
pageContext.getApplication();
JSP四大作用域
数据保存位置描述
Page只在一个页面中保存数据; javax.servlet.jsp.PageContext(抽象类)
Request只在一个请求中保存数据; javax.servlet.http.HttpServletRequest(接口)
Session在一次会话范围中保存数据,仅供单个用户使用;javax.servlet.http.HttpSession(接口)
Application在整个服务器上保存数据,所有用户共享;javax.servlet.ServletContext(接口)
<% 
pageContext.setAttribute("key",value); 
request.setAttribute("key",value); 
session.setAttribute("key",value); 
application.setAttribute("key",value); 
%>
JSP客户端请求

当浏览器请求一个网页时,它会向网络服务器发送一系列不能被直接读取的信息,因为这些信息是作为HTTP信息头的一部分来传送的。可以查阅HTTP协议来获得更多的信息。

消息头常见内容描述
Accept指定浏览器或其他客户端可以处理的MIME类型。它的值通常为 image/pngimage/jpeg
Accept-Charset指定浏览器要使用的字符集。比如 ISO-8859-1
Accept-Encoding指定编码类型。它的值通常为 gzipcompress
Accept-Language指定客户端首选语言,servlet会优先返回以当前语言构成的结果集,如果servlet支持这种语言的话。比如 en,en-us,ru等等
Connection表明客户端是否可以处理HTTP持久连接。持久连接允许客户端或浏览器在一个请求中获取多个文件。Keep-Alive 表示启用持久连接
Content-Length仅适用于POST请求,表示 POST 数据的字节数
Cookie返回先前发送给浏览器的cookies至服务器
Host指出原始URL中的主机名和端口号
If-Modified-Since表明只有当网页在指定的日期被修改后客户端才需要这个网页。 服务器发送304码给客户端,表示没有更新的资源
If-Unmodified-Since与If-Modified-Since相反, 只有文档在指定日期后仍未被修改过,操作才会成功
Referer标志着所引用页面的URL。比如,如果你在页面1,然后点了个链接至页面2,那么页面1的URL就会包含在浏览器请求页面2的信息头中
User-Agent用来区分不同浏览器或客户端发送的请求,并对不同类型的浏览器返回不同的内容
request对象常用方法描述
Cookie[] getCookies()返回客户端所有的Cookie的数组
Enumeration getAttributeNames()返回request对象的所有属性名称的集合
Enumeration getHeaderNames()返回所有HTTP头的名称集合
Enumeration getParameterNames()返回请求中所有参数的集合
HttpSession getSession()返回request对应的session对象,如果没有,则创建一个
HttpSession getSession(boolean create)返回request对应的session对象,如果没有并且参数create为true,则返回一个新的session对象
Locale getLocale()返回当前页的Locale对象,可以在response中设置
Object getAttribute(String name)返回名称为name的属性值,如果不存在则返回null。
ServletInputStream getInputStream()返回请求的输入流
String getAuthType()返回认证方案的名称,用来保护servlet,比如 “BASIC” 或者 “SSL” 或 null 如果 JSP没设置保护措施
String getCharacterEncoding()返回request的字符编码集名称
String getRemoteAddr()返回客户端的IP地址
int getIntHeader(String name)返回指定名称的request信息头的值
String getQueryString()返回此 request URL包含的查询字符串
String getProtocol()返回任何额外的与此request URL相关的路径
String getMethod()返回此request中的HTTP方法,比如 GET,,POST,或PUT
<%@ page import="java.io.*,java.util.*" %>
<html>
<head>
<title>HTTP Header Request Example</title>
</head>
<body>
<center>
<h2>HTTP Header Request Example</h2>
<table width="100%" border="1" align="center">
<tr bgcolor="#949494">
<th>Header Name</th><th>Header Value(s)</th>
</tr>
<%    
	Enumeration headerNames = request.getHeaderNames();    			  
	while(headerNames.hasMoreElements()) {       
		String paramName = (String)headerNames.nextElement();       
		out.print("<tr><td>" + paramName + "</td>\n");
        String paramValue = request.getHeader(paramName);
        out.println("<td> " + paramValue + "</td></tr>\n");
   }
%>
</table>
</center>
</body>
</html>
JSP服务器响应

Response响应对象主要将JSP容器处理后的结果传回到客户端。可以通过response变量设置HTTP的状态和向客户端发送数据,如Cookie、HTTP文件头信息等。

响应头常见内容描述
Allow指定服务器支持的request方法(GET,POST等等)
Cache-Control指定响应文档能够被安全缓存的情况。通常取值为 publicprivateno-cache 等等。 Public意味着文档可缓存,Private意味着文档只为单用户服务并且只能使用私有缓存。No-cache 意味着文档不被缓存。
Connection命令浏览器是否要使用持久的HTTP连接。close****值 命令浏览器不使用持久HTTP连接,而keep-alive 意味着使用持久化连接。
Content-Disposition让浏览器要求用户将响应以给定的名称存储在磁盘中
Content-Encoding指定传输时页面的编码规则
Expires指明啥时候过期并从缓存中移除
Last-Modified指明文档最后修改时间。客户端可以 缓存文档并且在后续的请求中提供一个 If-Modified-Since请求头
Refresh指明浏览器每隔多久请求更新一次页面。
Set-Cookie指明当前页面对应的cookie
Location在300秒内,包含所有的有一个状态码的响应地址,浏览器会自动重连然后检索新文档
Retry-After与503 (Service Unavailable)一起使用来告诉用户多久后请求将会得到响应
response常用方法描述
String encodeRedirectURL(String url)对sendRedirect()方法使用的URL进行编码
String encodeURL(String url)将URL编码,回传包含Session ID的URL
boolean containsHeader(String name)返回指定的响应头是否存在
boolean isCommitted()返回响应是否已经提交到客户端
void addCookie(Cookie cookie)添加指定的cookie至响应中
void addDateHeader(String name, long date)添加指定名称的响应头和日期值
void addHeader(String name, String value)添加指定名称的响应头和值
void flushBuffer()将任何缓存中的内容写入客户端
void reset()清除任何缓存中的任何数据,包括状态码和各种响应头
void resetBuffer()清除基本的缓存数据,不包括响应头和状态码
void sendError(int sc, String msg)使用指定的状态码和消息向客户端发送一个出错响应
void sendRedirect(String location)使用指定的URL向客户端发送一个临时的间接响应
void setBufferSize(int size)设置响应体的缓存区大小
void setCharacterEncoding(String charset)指定响应的编码集(MIME字符集),例如UTF-8
void setContentType(String type)设置响应的内容的类型,如果响应还未被提交的话
void setHeader(String name, String value)使用指定名称和值设置响应头的名称和内容
void setLocale(Locale loc)设置响应的语言环境,如果响应尚未被提交的话
void setStatus(int sc)设置响应的状态码
<%@ page import="java.io.*,java.util.*" %>
<html>
<head>
<title>Auto Refresh Header Example</title>
</head>
<body>
<center>
<h2>Auto Refresh Header Example</h2>
<%    
	// 设置每隔5秒自动刷新    
	response.setIntHeader("Refresh", 5);    
	// 获取当前时间    
	Calendar calendar = new GregorianCalendar();   
	String am_pm;
	int hour = calendar.get(Calendar.HOUR);    
	int minute = calendar.get(Calendar.MINUTE);    
	int second = calendar.get(Calendar.SECOND);    
	if(calendar.get(Calendar.AM_PM) == 0)       
		am_pm = "AM";    
	else       
		am_pm = "PM";    
	String CT = hour+":"+ minute +":"+ second +" "+ am_pm;    
	out.println("Current Time is: " + CT + "\n"); 
%>
</center>
</body>
</html>
常见HTTP状态码消息描述
100Continue只有一部分请求被服务器接收,但只要没被服务器拒绝,客户端就会延续这个请求
200OK请求被确认
201Created请求已完成,新的资源被创建
202Accepted请求被接受,但未处理完
301Moved Permanently被请求的页面已经移动到了新的URL下
302Found被请求的页面暂时性地移动到了新的URL下
400Bad Request服务器无法识别请求
404Not Found服务器无法找到所请求的页面
403Forbidden禁止访问所请求的页面
408Request Timeout请求时间超过了服务器所能等待的时间,连接被断开
500Internal Server Error请求不完整,服务器遇见了出乎意料的状况
503Service Unavailable请求不完整,服务器暂时重启或关闭
505HTTP Version Not Supported服务器不支持所指定的HTTP版本
设置HTTP状态码的方法描述
public void setStatus ( int statusCode )此方法可以设置任意的状态码。如果您的响应包含一个特殊的状态码和一个文档,请确保在用PrintWriter返回任何内容前调用setStatus方法
public void sendRedirect(String url)此方法产生302响应,同时产生一个 Location 头告诉URL 一个新的文档
public void sendError(int code, String message)此方法将一个状态码(通常为 404)和一个短消息,自动插入HTML文档中并发回给客户端
<html>
<head>
<title>Setting HTTP Status Code</title>
</head>
<body>
<%    // 设置错误代码,并说明原因    response.sendError(407, "Need authentication!!!" ); %>
</body>
</html>
JSP表单处理

我们在浏览网页的时候,经常需要向服务器提交信息,并让后台程序处理。浏览器中使用 GET 和 POST 方法向服务器提交数据。

  • GET方法

GET方法将请求的编码信息添加在网址后面,网址与编码信息通过"?"号分隔。如下所示:

//www.w3cschool.cn/hello?key1=value1&key2=value2

GET方法是浏览器默认传递参数的方法,一些敏感信息,如密码等建议不使用GET方法。

用get时,传输数据的大小有限制 (注意不是参数的个数有限制),最大为1024字节。

  • POST方法

一些敏感信息,如密码等我们可以同过POST方法传递,post提交数据是隐式的

POST提交数据是不可见的,GET是通过在url里面传递的(可以看一下你浏览器的地址栏)。

JSP使用getParameter()来获得传递的参数,getInputStream()方法用来处理客户端的二进制数据流的请求。

  • 读取表单数据
方法描述
getParameter()使用 request.getParameter() 方法来获取表单参数的值。
getParameterValues()接收数组变量 ,如checkobx类型
getParameterNames()取得所有变量的名称,该方法返回一个Emumeration。
getInputStream()读取来自客户端的二进制数据流。
  • 实例
<html>
<head>
<title>Using GET and POST Method to Read Form Data</title>
</head>
<body>
<center>
<h1>Using GET Method to Read Form Data</h1>
<ul>
<li><p><b>First Name:</b>
   <%= request.getParameter("first_name")%>
</p></li>
<li><p><b>Last  Name:</b>
   <%= request.getParameter("last_name")%>
</p></li>
</ul>
</body>
</html>
<html>
<body>
<form action="main.JSP" method="POST">
First Name: <input type="text" name="first_name">
<br />
Last Name: <input type="text" name="last_name" />
<input type="submit" value="Submit" />
</form>
</body>
</html>
JSP过滤器

Servlet和JSP中的过滤器都是Java类,主要用于:在请求访问后端资源时拦截它、管理从服务器返回给客户端的响应。

有认证过滤器、数据压缩过滤器、加密过滤器、触发资源访问事件的过滤器、登录和验证过滤器、MIME类型链过滤器、令牌过滤器等。

  • 过程
  1. 过滤器将会被插入进web.xml文件中,然后映射servlet、JSP文件的名字或URL模式。

  2. 当JSP容器启动网络应用程序时,它会创建每一个过滤器的实例,这些过滤器必须在部署描述文件web.xml中声明,并且按声明的顺序执行。

  • 过滤器方法
方法描述
public void doFilter (ServletRequest, ServletResponse, FilterChain)每当 request/response要通过过滤链时容器会调用这个方法,因为客户端请求链尾的资源
public void init(FilterConfig filterConfig)表明一个过滤器被安置在服务中
public void destroy()表明一个过滤器正在从服务中移除
  • 实例
/*
* 这个例子将会打印IP地址和每次访问JSP文件的日期时间。
*/
//  引入Java包
import java.io.*;
import javax.servlet.*;
import javax.servlet.http.*;
import java.util.*;
 
// 实现 Filter 类
public class LogFilter implements Filter  {
   public void  init(FilterConfig config) 
                         throws ServletException{
      // 获取初始化参数
      String testParam = config.getInitParameter("test-param"); 
 
      //打印初始化参数
      System.out.println("Test Param: " + testParam); 
   }
   public void  doFilter(ServletRequest request, 
                 ServletResponse response,
                 FilterChain chain) 
                 throws java.io.IOException, ServletException {
 
      // 获取客户端ip地址  
      String ipAddress = request.getRemoteAddr();
 
      // 输出ip地址及当前时间
      System.out.println("IP "+ ipAddress + ", Time "
                                       + new Date().toString());
 
      // 传递请求道过滤器链
      chain.doFilter(request,response);
   }
   public void destroy( ){
      /* 在Filter实例在服务器上被移除前调用。*/
   }
}
//web.xml文件中的映射
<filter>
   <filter-name>LogFilter</filter-name>
   <filter-class>LogFilter</filter-class>
   <init-param>
	  <param-name>test-param</param-name>
	  <param-value>Initialization Paramter</param-value>
   </init-param>
</filter>
<filter-mapping>
   <filter-name>LogFilter</filter-name>
   <url-pattern>/*</url-pattern>
</filter-mapping>
JSP Cookies处理
  • Cookies传递
  1. 服务器脚本发送一系列cookies至浏览器。比如名字,年龄,ID号码等等。

  2. 浏览器在本地机中存储这些信息,以备不时之需。

  3. 当下一次浏览器发送任何请求至服务器时,它会同时将这些cookies信息发送给服务器,然后服务器使用这些信息来识别用户或者干些其它事情。

  • 剖析

Cookies通常在HTTP信息头中设置(虽然JavaScript能够直接在浏览器中设置cookies)。在JSP中,设置一个cookie需要发送如下的信息头给服务器:

HTTP/1.1 200 OK
Date: Fri, 04 Feb 2000 21:03:38 GMT
Server: Apache/1.3.9 (UNIX) PHP/4.0b3
Set-Cookie: name=xyz; expires=Friday, 04-Feb-07 22:03:38 GMT; 
                 path=/; domain=tutorialspoint.com
Connection: close
Content-Type: text/html

Set-Cookie信息头包含一个键值对,一个GMT(格林尼治标准)时间,一个路径,一个域名。键值对会被编码为URL。

如果浏览器被配置成可存储cookies,那么它将会保存这些信息直到过期。如果用户访问的任何页面匹配了cookie中的路径和域名,那么浏览器将会重新将这个cookie发回给服务器。浏览器端的信息头长得就像下面这样:

GET / HTTP/1.0
Connection: Keep-Alive
User-Agent: Mozilla/4.6 (X11; I; Linux 2.2.6-15apmac ppc)
Host: zink.demon.co.uk:1126
Accept: image/gif, */*
Accept-Encoding: gzip
Accept-Language: en
Accept-Charset: iso-8859-1,*,utf-8
Cookie: name=xyz

JSP脚本通过request对象中的getCookies()方法来访问这些cookies,这个方法会返回一个Cookie对象的数组。

  • 常用方法
方法描述
public void setDomain(String pattern)设置cookie的域名
public String getDomain()获取cookie的域名
public void setMaxAge(int expiry)设置cookie有效期,以秒为单位,默认有效期为当前session的存活时间
public int getMaxAge()获取cookie有效期,以秒为单位,默认为-1 ,表明cookie会活到浏览器关闭为止
public String getName()返回 cookie的名称,名称创建后将不能被修改
public void setValue(String newValue)设置 cookie的值
public String getValue()获取cookie的值
public void setPath(String uri)设置cookie 的路径,默认为当前页面目录下的所有URL,还有此目录下的所有子目录
public String getPath()获取cookie 的路径
public void setSecure(boolean flag)指明cookie是否要加密传输
public void setComment(String purpose)设置注释描述 cookie的目的。当浏览器将cookie展现给用户时,注释将会变得非常有用
public String getComment()返回描述cookie目的的注释,若没有则返回null
  • 使用步骤
  1. 创建Cookies对象
Cookie cookie = new Cookie("key","value");
  1. 设置有效期
cookie.setMaxAge(60*60*24); 
  1. 发送到HTTP响应头
response.addCookie(cookie);
  • 实例
<%    
  // 为 first_name 和 last_name设置cookie          
  Cookie firstName = new Cookie("first_name", request.getParameter("first_name"));    
  Cookie lastName = new Cookie("last_name",request.getParameter("last_name"));     
  // 设置cookie过期时间为24小时。    
  firstName.setMaxAge(60*60*24);     
  lastName.setMaxAge(60*60*24);      
  // 在响应头部添加cookie    
  response.addCookie( firstName );    
  response.addCookie( lastName ); 
%>
<html>
<head>
<title>Setting Cookies</title>
</head>
<body>
<center>
<h1>Setting Cookies</h1>
</center>
<ul>
<li><p><b>First Name:</b>
<%= request.getParameter("first_name")%>
</p></li>
<li><p><b>Last  Name:</b>
   <%= request.getParameter("last_name")%>
</p></li>
</ul>
</body>
</html>
<html>
<head>
<title>Reading Cookies</title>
</head>
<body>
<center>
<h1>Reading Cookies</h1>
</center>
<%    
  Cookie cookie = null;    
  Cookie[] cookies = null;    
  // 获取cookies的数据,是一个数组    
  cookies = request.getCookies();    
  if( cookies != null ){       
    out.println("<h2> Found Cookies Name and Value</h2>");
    for (int i = 0; i < cookies.length; i++){          
      cookie = cookies[i]; 
         
      out.print("Name : " + cookie.getName( ) + ",  ");          
      out.print("Value: " + cookie.getValue( )+" <br/>");
    }
  }else{
      out.println("<h2>No cookies founds</h2>");
  }
%>
</body>
</html>
  • 删除Cookies
  1. 获取一个已经存在的cookie然后存储在Cookie对象中。

  2. 将cookie的有效期设置为0。

  3. 将这个cookie重新添加进响应头中。

JSP Session

HTTP是无状态协议,这意味着每次客户端检索网页时,都要单独打开一个服务器连接,因此服务器不会记录下先前客户端请求的任何信息。JSP利用servlet提供的HttpSession接口来识别一个用户,存储这个用户的所有访问信息。

有三种方法来维持客户端与服务器的会话:

  • Cookies

网络服务器可以指定一个唯一的session ID作为cookie来代表每个客户端,用来识别这个客户端接下来的请求。

这可能不是一种有效的方式,因为很多时候浏览器并不一定支持cookie,所以我们不建议使用这种方法来维持会话。

  • 隐藏表单域

一个网络服务器可以发送一个隐藏的HTML表单域和一个唯一的session ID,就像下面这样:

<input type="hidden" name="sessionid" value="12345">

这个条目意味着,当表单被提交时,指定的名称和值将会自动包含在GET或POST数据中。每当浏览器发送一个请求,session_id的值就可以用来保存不同浏览器的轨迹。

这种方式可能是一种有效的方式,点击<A HREF>标签中的超链接时不会产生表单提交事件,因此隐藏表单域也不支持通用会话跟踪。

  • 重写URL

您可以在每个URL后面添加一些额外的数据来区分会话,服务器能够根据这些数据来关联session标识符。

http://w3cschool.cn/file.htm;sessionid=12345

相比而言,重写URL是更好的方式来,就算浏览器不支持cookies也能工作,但缺点是您必须为每个URL动态指定session ID,就算这是个简单的HTML页面。

  • session对象

默认情况下,JSP允许会话跟踪,一个新的HttpSession对象将会自动地为新的客户端实例化。禁止会话跟踪需要显式地关掉它:

<%@ page session="false" %>
方法描述
public Object getAttribute(String name)返回session对象中与指定名称绑定的对象,如果不存在则返回null
public Enumeration getAttributeNames()返回session对象中所有的对象名称
public String getId()返回session对象的ID
public int getMaxInactiveInterval()返回最大时间间隔,以秒为单位,servlet 容器将会在这段时间内保持会话打开
public void setMaxInactiveInterval(int interval)用来指定时间,以秒为单位,servlet容器将会在这段时间内保持会话有效
public void removeAttribute(String name)移除session中指定名称的对象
public void invalidate()将session无效化,解绑任何与该session绑定的对象
public void setAttribute(String name, Object value)使用指定的名称和值来产生一个对象并绑定到session中
public long getLastAccessedTime()返回客户端最后访问的时间,以毫秒为单位,从1970年1月1号凌晨开始算起
  • 实例
<%@ page import="java.io.*,java.util.*" %>
<%  // 获取session创建时间
	Date createTime = new Date(session.getCreationTime());   
	// 获取最后访问页面的时间
    Date lastAccessTime = new Date(session.getLastAccessedTime());
    String title = "Welcome Back to my website";
    Integer visitCount = new Integer(0);
    String visitCountKey = new String("visitCount");
    String userIDKey = new String("userID");
    String userID = new String("ABCD");
    // 检测网页是否由新的访问用户    
    if (session.isNew()){
    	title = "Welcome to my website";
    	session.setAttribute(userIDKey, userID);
    	session.setAttribute(visitCountKey,  visitCount);
    }
    visitCount = (Integer)session.getAttribute(visitCountKey);
    visitCount = visitCount + 1;
    userID = (String)session.getAttribute(userIDKey); 	       
    session.setAttribute(visitCountKey,  visitCount); 
%>
<html>
<head>
<title>Session Tracking</title>
</head>
<body>

</body>
</html>
  • 删除Session数据
选择方法
移除一个特定的属性调用public void removeAttribute(String name)
删除整个会话调用public void invalidate()
设置会话有效期调用 public void setMaxInactiveInterval(int interval)方法来设置session超时
登出用户支持servlet2.4版本的服务器,可以调用 logout()方法来登出用户,并且使所有相关的session无效。
配置web.xml文件如果使用的是Tomcat,可以配置web.xml文件来设置超时,Tomcat中以分钟为单位,默认30分钟
<session-config>
    <session-timeout>15</session-timeout>
</session-config>
JSP文件上传

JSP可以通过HTML的form表单上传文件到服务器。 文件类型可以是文本文件、二进制文件、图像文件等其他任何文档。

  • 文件上传表单
<html>
<head>
<title>File Uploading Form</title>
</head>
<body>
<h3>File Upload:</h3>
Select a file to upload: <br />
<form action="UploadServlet" method="post"
                        enctype="multipart/form-data">
<input type="file" name="file" size="50" />
<br />
<input type="submit" value="Upload File" />
</form>
</body>
</html>
  • Xml配置映射

首先我们先定义文件上传后存储在服务上的位置,你可以将路径写在你的程序当中,或者我们可以在web.xml配置文件中通过设置 context-param 元素来设置文件存储的目录,如下所示:

<web-app>
....
<context-param> 
    <description>文件上传地址</description> 
    <param-name>file-upload</param-name> 
    <param-value>
         c:\apache-tomcat-5.5.29\webapps\data\
     </param-value> 
</context-param>
....
</web-app>
  • JSP处理脚本
  1. 以下实例依赖FileUpload, 所以你需要在你的classpath中引入最新的 commons-fileupload.x.x.jar 包文件。 下载地址为:http://commons.apache.org/fileupload/
  2. FileUpload 依赖 Commons IO, 所以你需要在你的classpath中引入最新的 commons-io-x.x.jar 。 下载地址为:http://commons.apache.org/io/
  3. 确保你已经创建了目录 c:\temp 和 c:\apache-tomcat-5.5.29\webapps\data
<%@ page import="java.io.*,java.util.*, javax.servlet.*" %>
<%@ page import="javax.servlet.http.*" %>
<%@ page import="org.apache.commons.fileupload.*" %>
<%@ page import="org.apache.commons.fileupload.disk.*" %>
<%@ page import="org.apache.commons.fileupload.servlet.*" %>
<%@ page import="org.apache.commons.io.output.*" %>

<%
   File file ;
   int maxFileSize = 5000 * 1024;
   int maxMemSize = 5000 * 1024;
   ServletContext context = pageContext.getServletContext();
   String filePath = context.getInitParameter("file-upload");

   // 验证上传内容了类型
   String contentType = request.getContentType();
   if ((contentType.indexOf("multipart/form-data") >= 0)) {

      DiskFileItemFactory factory = new DiskFileItemFactory();
      // 设置内存中存储文件的最大值
      factory.setSizeThreshold(maxMemSize);
      // 本地存储的数据大于 maxMemSize.
      factory.setRepository(new File("c:\\temp"));

      // 创建一个新的文件上传处理程序
      ServletFileUpload upload = new ServletFileUpload(factory);
      // 设置最大上传的文件大小
      upload.setSizeMax( maxFileSize );
      try{ 
         // 解析获取的文件
         List fileItems = upload.parseRequest(request);

         // 处理上传的文件
         Iterator i = fileItems.iterator();

         out.println("<html>");
         out.println("<head>");
         out.println("<title>JSP File upload</title>");  
         out.println("</head>");
         out.println("<body>");
         while ( i.hasNext () ) 
         {
            FileItem fi = (FileItem)i.next();
            if ( !fi.isFormField () )	
            {
            // 获取上传文件的参数
            String fieldName = fi.getFieldName();
            String fileName = fi.getName();
            boolean isInMemory = fi.isInMemory();
            long sizeInBytes = fi.getSize();
            // 写入文件
            if( fileName.lastIndexOf("\\") >= 0 ){
            file = new File( filePath , 
            fileName.substring( fileName.lastIndexOf("\\"))) ;
            }else{
            file = new File( filePath ,
            fileName.substring(fileName.lastIndexOf("\\")+1)) ;
            }
            fi.write( file ) ;
            out.println("Uploaded Filename: " + filePath + 
            fileName + "<br>");
            }
         }
         out.println("</body>");
         out.println("</html>");
      }catch(Exception ex) {
         System.out.println(ex);
      }
   }else{
      out.println("<html>");
      out.println("<head>");
      out.println("<title>Servlet upload</title>");  
      out.println("</head>");
      out.println("<body>");
      out.println("<p>No file uploaded</p>"); 
      out.println("</body>");
      out.println("</html>");
   }
%>
JSP日期处理

由于JSP可以使用所有Java API,直接使用Java中的Date类即可,在java.util包下。

  • 常用方法
方法描述
boolean after(Date date)如果比给定的日期晚,则返回true,否则返回false
boolean before(Date date)如果比给定的日期早,则返回true,否则返回false
int compareTo(Date date)如果与给定日期相等,则返回0,如果比给定日期早,则返回一个负数,如果比给定日期晚,则返回一个正数
boolean equals(Object date)如果与给定日期相同,则返回true,否则返回false
long getTime( )返回从1970年1月1日凌晨至此对象所表示时间的毫秒数
void setTime(long time)使用给定参数设置时间和日期,参数time表示从1970年1月1日凌晨至time所经过的毫秒数
String toString( )将此对象转换为字符串并返回这个字符串
  • SimpleDateFormat格式化日期
<%
	Date dNow = new Date( );
	SimpleDateFormat ft = new SimpleDateFormat ("E yyyy.MM.dd 'at' hh:mm:ss a zzz");
	out.print( "<h2 align=\"center\">" + ft.format(dNow) + "</h2>");
%>

输出:

Display Current Date & Time
Mon 2013.06.21 at 10:06:44 PM GMT+04:00
  • SimpleDateFormat格式码
字符描述示例
G时代标识符AD
y4位数年份2001
MJuly or 07
d10
h12小时制, A.M./P.M. (1~12)12
H24小时制22
m分钟30
s55
S毫秒234
E星期Tuesday
D一年中的某天360
F一个月中第几个当日星期2 (second Wed. in July)
w一年中的某星期40
W一个月中的某星期1
aA.M./P.M. 标记PM
k一天中的某个小时 (1~24)24
K一天中的某个小时,A.M./P.M. (0~11)10
z时区Eastern Standard Time
文本分隔Delimiter
"单引号`
JSP重定向
  • 使用response对象的sendRedirect()方法
public void response.sendRedirect(String location)
throws IOException 
  • 使用setStatus()和setHeader()方法
<%
	// 重定向到新地址
	String site = new String("http://www.w3cschool.cn");    						response.setStatus(response.SC_MOVED_TEMPORARILY);    							response.setHeader("Location", site);
%>
JSP邮件

使用JSP实现邮件发送功能很简单,需要有JavaMail API,并且需要安装JavaBean Activation Framework。

  • 发送简单邮件

这个例子展示了如何从您的机器发送一封简单的邮件。它假定localhost已经连接至网络并且有能力发送一封邮件。与此同时,请再一次确认mail.jar包和activation.jar包已经添加进CLASSPATH变量中。

<%@ page import="java.io.*,java.util.*,javax.mail.*"%>
<%@ page import="javax.mail.internet.*,javax.activation.*"%>
<%@ page import="javax.servlet.http.*,javax.servlet.*" %>
<%  
	String result;    
	// 收件人的电子邮件
	String to = "abcd@gmail.com";     
	// 发件人的电子邮件   
	String from = "mcmohd@gmail.com";   
	// 假设你是从本地主机发送电子邮件  
	String host = "localhost";  
	// 获取系统属性对象   
	Properties properties = System.getProperties();    
	// 设置邮件服务器  
	properties.setProperty("mail.smtp.host", host);   
	// 获取默认的Session对象。  
	Session mailSession = Session.getDefaultInstance(properties);  
	try{     
    	// 创建一个默认的MimeMessage对象。    
   	 	MimeMessage message = new MimeMessage(mailSession);   
    	// 设置 From: 头部的header字段    
   		message.setFrom(new InternetAddress(from));   
    	// 设置 To: 头部的header字段       
    	message.addRecipient(Message.RecipientType.TO, new InternetAddress(to)); 
    	// 设置 Subject: header字段      
    	message.setSubject("This is the Subject Line!");    
    	// 现在设置的实际消息  
    	message.setText("This is actual message");    
    	// 发送消息    
    	Transport.send(message);   
    	result = "Sent message successfully....";    }
	catch (MessagingException mex) {    
    	mex.printStackTrace();       
    	result = "Error: unable to send message....";  
		} 
%>
<html>
<head>
<title>Send Email using JSP</title>
</head>
<body>
<center>
<h1>Send Email using JSP</h1>
</center>
<p align="center">
<%     out.println("Result: " + result + "\n"); %>
</p>
</body>
</html>

如果想要把邮件发送给多人,下面列出的方法可以用来指明多个邮箱地址:

void addRecipients(Message.RecipientType type, 
                   Address[] addresses)
throws MessagingException

参数的描述如下:

type:这个值将会被设置成TO,CC,或BCC。CC代表副本,BCC代表黑色副本,例子程序中使用的是TO。

addresses:这是一个邮箱地址的数组,当指定邮箱地址时需要使用InternetAddress()方法。

  • 发送HTML邮件

这个例子和前一个例子非常相似,不过在这个例子中我们使用了setContent()方法,将"text/html"做为第二个参数传给它,用来表明消息中包含了HTML内容。

<%   
	String result;   
	// 收件人的电子邮件   
	String to = "abcd@gmail.com";    
	// 发件人的电子邮件  
	String from = "mcmohd@gmail.com";   
	// 假设你是从本地主机发送电子邮件   
	String host = "localhost";   
	// 获取系统属性对象   
	Properties properties = System.getProperties();  
	// 设置邮件服务器   
	properties.setProperty("mail.smtp.host", host);    
	// 获取默认的Session对象。  
	Session mailSession = Session.getDefaultInstance(properties);  
	try{   
		// 创建一个默认的MimeMessage对象。    
    	MimeMessage message = new MimeMessage(mailSession);    
    	// 设置 From: 头部的header字段  
    	message.setFrom(new InternetAddress(from));     
    	// 设置 To: 头部的header字段     
    	message.addRecipient(Message.RecipientType.TO, new InternetAddress(to)); 
    	// 设置 Subject: header字段     
    	message.setSubject("This is the Subject Line!");       
    	// 设置 HTML消息     
    	message.setContent("<h1>This is actual message</h1>", "text/html" );
      	// 发送消息
      	Transport.send(message);
      	result = "Sent message successfully....";
   }catch (MessagingException mex) {
      	mex.printStackTrace();
      	result = "Error: unable to send message....";
   }
%>
  • 发送附件邮件
<%    
String result;    
// 收件人的电子邮件    
String to = "abcd@gmail.com";     
// 发件人的电子邮件    
String from = "mcmohd@gmail.com";     
// 假设你是从本地主机发送电子邮件    
String host = "localhost";    
// 获取系统属性对象   
Properties properties = System.getProperties();    
// 设置邮件服务器    
properties.setProperty("mail.smtp.host", host);   
// 获取默认的Session对象。    
Session mailSession = Session.getDefaultInstance(properties);     
try{      
// 创建一个默认的MimeMessage对象。      
    MimeMessage message = new MimeMessage(mailSession);        
    // 设置 From: 头部的header字段       
    message.setFrom(new InternetAddress(from));        
    // 设置 To: 头部的header字段       	
    message.addRecipient(Message.RecipientType.TO, new InternetAddress(to));     
    // 设置 Subject: header字段
    message.setSubject("This is the Subject Line!");        
    // 创建消息部分
    BodyPart messageBodyPart = new MimeBodyPart();
    // 填充消息
    messageBodyPart.setText("This is message body");
    // 创建多媒体消息
    Multipart multipart = new MimeMultipart();
    // 设置文本消息部分       
    multipart.addBodyPart(messageBodyPart);   
    // 附件部分     
    messageBodyPart = new MimeBodyPart();   
    String filename = "file.txt";    
    DataSource source = new FileDataSource(filename);       						messageBodyPart.setDataHandler(new DataHandler(source));       					messageBodyPart.setFileName(filename);       									multipart.addBodyPart(messageBodyPart);       
    // 发送完整消息    
    message.setContent(multipart );   
    // 发送消息     
    Transport.send(message);   
    String title = "Send Email";    
    result = "Sent message successfully....";    }
catch (MessagingException mex) {      
    mex.printStackTrace();       
    result = "Error: unable to send message....";    } 
%>
  • 用户认证
 props.setProperty("mail.user", "myuser");
 props.setProperty("mail.password", "mypwd");
  • 使用表单发送邮件
String to = request.getParameter("to");
String from = request.getParameter("from");
String subject = request.getParameter("subject");
String messageText = request.getParameter("body");

JSP高级教程

JSP Standard Tag Library

JSP标准标签库(JSTL)是一个JSP标签集合,它封装了JSP应用的通用核心功能。

JSTL支持通用的、结构化的任务,比如迭代,条件判断,XML文档操作,国际化标签,SQL标签。

根据JSTL标签所提供的功能,可以将其分为5个类别:核心标签格式化标签SQL 标签XML 标签JSTL 函数

  • JSTL库安装

Apache Tomcat安装JSTL 库步骤如下:

  1. 从Apache的标准标签库中下载的二进包(jakarta-taglibs-standard-current.zip)。下载地址:http://archive.apache.org/dist/jakarta/taglibs/standard/binaries/

  2. 下载jakarta-taglibs-standard-1.1.1.zip 包并解压,将jakarta-taglibs-standard-1.1.1/lib/下的两个jar文件:standard.jar和jstl.jar文件拷贝到/WEB-INF/lib/下。

  3. 接下来我们在 web.xml 文件中添加以下配置,使用哪个库就加哪个配置就行。

	<JSP-config>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/fmt</taglib-uri>
	<taglib-location>/WEB-INF/fmt.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/fmt-rt</taglib-uri>
	<taglib-location>/WEB-INF/fmt-rt.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/core</taglib-uri>
	<taglib-location>/WEB-INF/c.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/core-rt</taglib-uri>
	<taglib-location>/WEB-INF/c-rt.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/sql</taglib-uri>
	<taglib-location>/WEB-INF/sql.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/sql-rt</taglib-uri>
	<taglib-location>/WEB-INF/sql-rt.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/x</taglib-uri>
	<taglib-location>/WEB-INF/x.tld</taglib-location>
	</taglib>
	<taglib>
	<taglib-uri>http://java.sun.com/jstl/x-rt</taglib-uri>
	<taglib-location>/WEB-INF/x-rt.tld</taglib-location>
	</taglib>
	</jsp-config>
  • 核心标签
<%@ taglib prefix="c"            
           uri="http://java.sun.com/jsp/jstl/core" %>
标签描述
<c:out>用于在JSP中显示数据,就像<%= … >
<c:set>用于保存数据
<c:remove>用于删除数据
<c:catch>用来处理产生错误的异常状况,并且将错误信息储存起来
<c:if>与我们在一般程序中用的if一样
<c:choose>本身只当做<c:when><c:otherwise>的父标签
<c:when><c:choose>的子标签,用来判断条件是否成立
<c:otherwise><c:choose>的子标签,接在<c:when>标签后,当<c:when>标签判断为false时被执行
<c:import>检索一个绝对或相对 URL,然后将其内容暴露给页面
<c:forEach>基础迭代标签,接受多种集合类型
<c:forTokens>根据指定的分隔符来分隔内容并迭代输出
<c:param>用来给包含或重定向的页面传递参数
<c:redirect>重定向至一个新的URL.
<c:url>使用可选的查询参数来创造一个URL
  • 格式化标签
<%@ taglib prefix="fmt" 
           uri="http://java.sun.com/jsp/jstl/fmt" %>
标签描述
fmt:formatNumber使用指定的格式或精度格式化数字
fmt:parseNumber解析一个代表着数字,货币或百分比的字符串
fmt:formatDate使用指定的风格或模式格式化日期和时间
fmt:parseDate解析一个代表着日期或时间的字符串
fmt:bundle绑定资源
fmt:setLocale指定地区
fmt:setBundle绑定资源
fmt:timeZone指定时区
fmt:setTimeZone指定时区
fmt:message显示资源配置文件信息
fmt:requestEncoding设置request的字符编码
  • SQL标签
<%@ taglib prefix="sql"             
           uri="http://java.sun.com/jsp/jstl/sql" %>
标签描述
sql:setDataSource指定数据源
sql:query运行SQL查询语句
sql:update运行SQL更新语句
sql:param将SQL语句中的参数设为指定值
dateParam将SQL语句中的日期参数设为指定的java.util.Date 对象值
sql:transaction在共享数据库连接中提供嵌套的数据库行为元素,将所有语句以一个事务的形式来运行
  • XML标签
<%@ taglib prefix="x"             
           uri="http://java.sun.com/jsp/jstl/xml" %>
标签描述
<x:out>与<%= … >,类似,不过只用于XPath表达式
<x:parse>解析 XML 数据
<x:set>设置XPath表达式
<x:if>判断XPath表达式,若为真,则执行本体中的内容,否则跳过本体
<x:forEach>迭代XML文档中的节点
<x:choose><x:when><x:otherwise>的父标签
<x:when><x:choose>的子标签,用来进行条件判断
<x:otherwise><x:choose>的子标签,当<x:when>判断为false时被执行
<x:transform>将XSL转换应用在XML文档中
<x:param><x:transform>共同使用,用于设置XSL样式表
  • JSTL函数
<%@ taglib prefix="fn"             
           uri="http://java.sun.com/jsp/jstl/functions" %>
函数描述
fn:contains()测试输入的字符串是否包含指定的子串
fn:containsIgnoreCase()测试输入的字符串是否包含指定的子串,大小写不敏感
fn:endsWith()测试输入的字符串是否以指定的后缀结尾
fn:escapeXml()跳过可以作为XML标记的字符
fn:indexOf()返回指定字符串在输入字符串中出现的位置
fn:join()将数组中的元素合成一个字符串然后输出
fn:length()返回字符串长度
fn:replace()将输入字符串中指定的位置替换为指定的字符串然后返回
fn:split()将字符串用指定的分隔符分隔然后组成一个子字符串数组并返回
fn:startsWith()测试输入字符串是否以指定的前缀开始
fn:substring()返回字符串的子集
fn:substringAfter()返回字符串在指定子串之后的子集
fn:substringBefore()返回字符串在指定子串之前的子集
fn:toLowerCase()将字符串中的字符转为小写
fn:toUpperCase()将字符串中的字符转为大写
fn:trim()移除首位的空白符
JSP数据库
  • 连接数据库
<sql:setDataSource var="snapshot" driver="com.mysql.cj.jdbc.Driver"
     url="jdbc:mysql://localhost:3306/db_book?serverTimezone=UTC"
     user="root"  password="1234"/>
  • select操作
<sql:query dataSource="${snapshot}" var="result">
SELECT * from Employees;
</sql:query>
<table border="1" width="100%">
<tr>
   <th>Emp ID</th>
   <th>First Name</th>
   <th>Last Name</th>
   <th>Age</th>
</tr>
<c:forEach var="row" items="${result.rows}">
<tr>
   <td><c:out value="${row.id}"/></td>
   <td><c:out value="${row.first}"/></td>
   <td><c:out value="${row.last}"/></td>
   <td><c:out value="${row.age}"/></td>
</tr>
</c:forEach>
</table>
  • insert操作
<sql:update dataSource="${snapshot}" var="result">
INSERT INTO Employees VALUES (104, 2, 'Nuha', 'Ali');
</sql:update>
  • delete操作
<sql:update dataSource="${snapshot}" var="count">
  DELETE FROM Employees WHERE Id = ?
  <sql:param value="${empId}" />
</sql:update>
  • update操作
<sql:update dataSource="${snapshot}" var="count">
  UPDATE Employees SET last = 'Ali' WHERE Id = ?
  <sql:param value="${empId}" />
</sql:update>
JSP XML 数据处理

当通过HTTP发送XML数据时,就有必要使用JSP来处理传入和流出的XML文档了,比如RSS文档。

  • 发送XML

使用JSP发送XML内容就和发送HTML内容一样。唯一的不同就是您需要把页面的context属性设置为text/xml。

<%@ page contentType="text/xml" %>

<books>
   <book>
      <name>Padam History</name>
      <author>ZARA</author>
      <price>100</price>
   </book>
</books>
  • 处理XML

在使用JSP处理XML之前,您需要将与XML 和XPath相关的两个库文件放在Tomcat的\lib目录下:

books.xml文件:

<books>
<book>
  <name>Padam History</name>
  <author>ZARA</author>
  <price>100</price>
</book>
<book>
  <name>Great Mistry</name>
  <author>NUHA</author>
  <price>2000</price>
</book>
</books>
<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ taglib prefix="x" uri="http://java.sun.com/jsp/jstl/xml" %>
 
<html>
<head>
  <title>JSTL x:parse Tags</title>
</head>
<body>
<h3>Books Info:</h3>
<c:import var="bookInfo" url="http://localhost:8080/books.xml"/>
 
<x:parse xml="${bookInfo}" var="output"/>
<b>The title of the first book is</b>: 
<x:out select="$output/books/book[1]/name" />
<br>
<b>The price of the second book</b>: 
<x:out select="$output/books/book[2]/price" />
 
</body>
</html>
  • 格式化XML

XSLT样式表style.xsl文件:

<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl= "http://www.w3.org/1999/XSL/Transform" version="1.0">
 
<xsl:output method="html" indent="yes"/>
 
<xsl:template match="/">
  <html>
  <body>
   <xsl:apply-templates/>
  </body>
  </html>
</xsl:template>
 
<xsl:template match="books">
  <table border="1" width="100%">
    <xsl:for-each select="book">
      <tr>
        <td>
          <i><xsl:value-of select="name"/></i>
        </td>
        <td>
          <xsl:value-of select="author"/>
        </td>
        <td>
          <xsl:value-of select="price"/>
        </td>
      </tr>
    </xsl:for-each>
  </table>
</xsl:template>
</xsl:stylesheet>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ taglib prefix="x" uri="http://java.sun.com/jsp/jstl/xml" %>
 
<html>
<head>
  <title>JSTL x:transform Tags</title>
</head>
<body>
<h3>Books Info:</h3>
<c:set var="xmltext">
  <books>
    <book>
      <name>Padam History</name>
      <author>ZARA</author>
      <price>100</price>
    </book>
    <book>
      <name>Great Mistry</name>
      <author>NUHA</author>
      <price>2000</price>
    </book>
  </books>
</c:set>
 
<c:import url="http://localhost:8080/style.xsl" var="xslt"/>
<x:transform xml="${xmltext}" xslt="${xslt}"/>
 
</body>
</html>
JSP JavaBean

JavaBean是特殊的Java类,是用Java语言写成的可重用组件,并且遵守JavaBeans API规范。

JavaBean与其它Java类相比而言独一无二的特征:

  1. 提供一个默认的无参构造函数。

  2. 需要被序列化并且实现了Serializable接口。

  3. 可能有一系列可读写属性。

  4. 可能有一系列的"getter"或"setter"方法。

  • JavaBean属性

一个JavaBean对象的属性应该是可访问的。这个属性可以是任意合法的Java数据类型,包括自定义Java类,可以是可读写,或只读,或只写,对应"getter"或"setter"方法。

public class StudentsBean implements java.io.Serializable
{
   private String firstName = null;
   private String lastName = null;
   private int age = 0;

   public StudentsBean() {
   }
   public String getFirstName(){
      return firstName;
   }
   public String getLastName(){
      return lastName;
   }
   public int getAge(){
      return age;
   }
   public void setFirstName(String firstName){
      this.firstName = firstName;
   }
   public void setLastName(String lastName){
      this.lastName = lastName;
   }
   public void setAge(Integer age){
      this.age = age;
   }
}
  • 使用JavaBeans

<JSP:useBean>标签可以在JSP中声明一个JavaBean,然后使用。声明后,JavaBean对象就成了脚本变量,可以通过脚本元素或其他自定义标签来访问。<JSP:useBean>标签的语法格式如下:

<JSP:useBean id="bean's name" scope="bean's scope" typeSpec/>

scope的值可以是page,request,session或application。

<JSP:useBean id="date" class="java.util.Date" /> 
<p>The date/time is <%= date %>
  • 添加和访问属性
<JSP:useBean id="id" class="bean's class" scope="bean's scope">
   <JSP:setProperty name="bean's id" property="property name"                       value="value"/>
   <JSP:getProperty name="bean's id" property="property name"/>
   ...........
</jsp:useBean>
JSP自定义标签

自定义标签是用户定义的JSP语言元素。当JSP页面包含一个自定义标签时将被转化为servlet,标签转化为对被 称为tag handler的对象的操作,即当servlet执行时Web container调用那些操作。

你可以继承SimpleTagSupport类并重写的doTag()方法来开发一个最简单的自定义标签。

  • 简单的标签

    效果如下:

<ex:Hello />
  1. 创建Tag处理Java类
package com.tutorialspoint;

import javax.servlet.JSP.tagext.*;
import javax.servlet.JSP.*;
import java.io.*;

public class HelloTag extends SimpleTagSupport {

  public void doTag() throws JSPException, IOException {
    //getJSPContext()获取当前的JSPContext对象
    JSPWriter out = getJSPContext().getOut();
    out.println("Hello Custom Tag!");
  }
}
  1. 编译以上类,并将其复制到环境变量CLASSPATH目录中。最后创建如下标签库:Tomcat目录下的webapps\ROOT\WEB-INF\custom.tld。
<taglib>
  <tlib-version>1.0</tlib-version>
  <JSP-version>2.0</jsp-version>
  <short-name>Example TLD</short-name>
  <tag>
    <name>Hello</name>
    <tag-class>com.tutorialspoint.HelloTag</tag-class>
    <body-content>empty</body-content>
  </tag>
</taglib>
  • 标签体

效果如下:

<ex:Hello>
   This is message body
</ex:Hello>
  1. 修改HelloTag类
package com.tutorialspoint;

import javax.servlet.JSP.tagext.*;
import javax.servlet.JSP.*;
import java.io.*;

public class HelloTag extends SimpleTagSupport {

   StringWriter sw = new StringWriter();
   public void doTag()
      throws JSPException, IOException
    {
       //invoke()将标签体的内容写到sw流中
       getJSPBody().invoke(sw);
       getJSPContext().getOut().println(sw.toString());
    }

}
  1. 修改TLD文件中的<body-content>
<taglib>
  <tlib-version>1.0</tlib-version>
  <JSP-version>2.0</jsp-version>
  <short-name>Example TLD with Body</short-name>
  <tag>
    <name>Hello</name>
    <tag-class>com.tutorialspoint.HelloTag</tag-class>
    <body-content>scriptless</body-content>
  </tag>
</taglib>
  • 自定义标签属性

效果如下:

<ex:Hello message="This is custom tag" />
  1. 在HelloTag中添加属性,并至少实现setter方法

  2. 在TLD文件中添加<attribute>元素

<taglib>
  <tlib-version>1.0</tlib-version>
  <JSP-version>2.0</jsp-version>
  <short-name>Example TLD with Body</short-name>
  <tag>
    <name>Hello</name>
    <tag-class>com.tutorialspoint.HelloTag</tag-class>
    <body-content>scriptless</body-content>
    <attribute>
       <name>message</name>
    </attribute>
  </tag>
</taglib>

还可以包含以下属性:

属性描述
name定义属性的名称。每个标签的是属性名称必须是唯一的。
required指定属性是否是必须的或者可选的,如果设置为false为可选。
rtexprvalue声明在运行表达式时,标签属性是否有效。
type定义该属性的Java类类型 。默认指定为 String
description描述信息
fragment如果声明了该属性,属性值将被视为一个 JSPFragment

多属性:

.....
    <attribute>
      <name>attribute_name1</name>
      <required>false</required>
      <type>java.util.Boolean</type>
      <fragment>false</fragment>
    </attribute>
    <attribute>
      <name>attribute_name2</name>
      <required>true</required>
      <type>java.util.Date</type>
    </attribute>
.....
JSP表达式语言

JSP表达式语言(EL)使得访问存储在JavaBean中的数据变得非常简单。JSP EL既可以用来创建算术表达式也可以用来创建逻辑表达式。

在JSP EL表达式内可以使用整型数,浮点数,字符串,常量true、false,还有null。

  • 使用EL

JSP EL允许您指定一个表达式来表示属性值。一个简单的表达式语法如下:

${expr}

想要停用对EL表达式的评估的话,需要使用page指令将isELIgnored属性值设为true:

<%@ page isELIgnored ="true|false" %>

这样,EL表达式就会被忽略。若设为false,则容器将会计算EL表达式。

  • EL操作符
操作符描述
.访问一个Bean属性或者一个映射条目
[]访问一个数组或者链表的元素
( )组织一个子表达式以改变优先级
+
-减或负
*
/ or div
% or mod取模
== or eq测试是否相等
!= or ne测试是否不等
< or lt测试是否小于
> or gt测试是否大于
<= or le测试是否小于等于
>= or ge测试是否大于等于
&& or and测试逻辑与
|| or or测试逻辑或
! or not测试取反
empty测试是否空值
  • EL函数

JSP EL允许您在表达式中使用函数。这些函数必须被定义在自定义标签库中。函数的使用语法如下:

${ns:func(param1, param2, ...)}

ns指的是命名空间(namespace),func指的是函数的名称,param1指的是第一个参数,param2指的是第二个参数,以此类推。比如,有函数fn:length,在JSTL库中定义,可以像下面这样来获取一个字符串的长度:

${fn:length("Get my length")}

要使用任何标签库中的函数,您需要将这些库安装在服务器中,然后使用标签在JSP文件中包含这些库。

  • EL隐含对象
隐含对象描述
pageScopepage 作用域
requestScoperequest 作用域
sessionScopesession 作用域
applicationScopeapplication 作用域
paramRequest 对象的参数,字符串
paramValuesRequest对象的参数,字符串集合
headerHTTP 信息头,字符串
headerValuesHTTP 信息头,字符串集合
initParam上下文初始化参数
cookieCookie值
pageContext当前页面的pageContext

param和paramValues对象用来访问参数值,通过使用request.getParameter方法和request.getParameterValues方法。param对象返回单一的字符串,而paramValues对象则返回一个字符串数组。

举例来说,访问一个名为order的参数,可以这样使用表达式: p a r a m . o r d e r , 或 者 {param.order},或者 param.order{param[“order”]}。

pageScope,requestScope,sessionScope,applicationScope变量用来访问存储在各个作用域层次的变量。

举例来说,如果您需要显式访问在applicationScope层的box变量,可以这样来访问:applicationScope.box。

header和headerValues对象用来访问信息头,通过使用 request.getHeader方法和request.getHeaders方法。header对象返回单一值,而headerValues则返回一个字符串数组。

举例来说,要访问一个名为user-agent的信息头,可以这样使用表达式: h e a d e r . u s e r − a g e n t , 或 者 {header.user-agent},或者 header.useragent{header[“user-agent”]}。

JSP异常处理

JSP代码中通常有以下几类异常:

检查型异常:检查型异常就是一个典型的用户错误或者一个程序员无法预见的错误。举例来说,如果一个文件将要被打开,但是无法找到这个文件,则一个异常被抛出。这些异常不能再编译期被简单地忽略。

运行时异常:一个运行时异常可能已经被程序员避免,这种异常在编译期将会被忽略。

错误:这里没有异常,但问题是它超出了用户或者程序员的控制范围。错误通常会在代码中被忽略,您几乎不能拿它怎么样。举例来或,栈溢出错误。这些错误都会在编译期被忽略。

  • 使用exception对象

exception对象是Throwable子类的一个实例,只在错误页面中可用。下表列出了Throwable类中一些重要的方法:

方法描述
public String getMessage()返回异常的信息。这个信息在Throwable构造函数中被初始化
public ThrowablegetCause()返回引起异常的原因,类型为Throwable对象
public String toString()返回类名
public void printStackTrace()将异常栈轨迹输出至System.err
public StackTraceElement [] getStackTrace()以栈轨迹元素数组的形式返回异常栈轨迹
public ThrowablefillInStackTrace()使用当前栈轨迹填充Throwable对象

JSP提供了可选项来为每个JSP页面指定错误页面。无论何时页面抛出了异常,JSP容器都会自动地调用错误页面。如果您想要将异常处理放在一个页面中,并且对不同的异常进行不同的处理,需要使用try…catch块

使用<%@page errorPage="XXXXX"%>指令指定一个错误页面。

错误页面实例如下,开头必须包含<%@ page isErrorPage="true" %>说明。

<%@ page isErrorPage="true" %>
<html>
<head>
<title>Show Error Page</title>
</head>
<body>
<h1>Opps...</h1>
<p>Sorry, an error occurred.</p>
<p>Here is the exception stack trace: </p>
<pre>
<% exception.printStackTrace(response.getWriter()); %>
  • 使用JSTL标签
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@page isErrorPage="true" %>
<html>
<head>
<title>Show Error Page</title>
</head>
<body>
<h1>Opps...</h1>
<table width="100%" border="1">
<tr valign="top">
<td width="40%"><b>Error:</b></td>
<td>${pageContext.exception}</td>
</tr>
<tr valign="top">
<td><b>URI:</b></td>
<td>${pageContext.errorData.requestURI}</td>
</tr>
<tr valign="top">
<td><b>Status code:</b></td>
<td>${pageContext.errorData.statusCode}</td>
</tr>
<tr valign="top">
<td><b>Stack trace:</b></td>
<td>
<c:forEach var="trace" items="${pageContext.exception.stackTrace}">
<p>${trace}</p>
</c:forEach>
</td>
</tr>
</table>
</body>
</html>
JSP调试
  • 使用System.out.println()
  1. System.out.println()可以很方便地标记一段代码是否被执行,也可以打印出各种各样的值。

  2. 自从System对象成为Java核心对象后,它便可以使用在任何地方而不用引入额外的类。使用范围包括Servlets,JSP,RMI,EJB’s,Beans,类和独立应用。

  3. 与在断点处停止运行相比,用System.out进行输出不会对应用程序的运行流程造成重大的影响,这个特点在定时机制非常重要的应用程序中就显得非常有用了。

  • JDB Logger
  1. J2SE日志框架可为任何运行在JVM中的类提供日志记录服务。因此我们可以利用这个框架来记录任何信息。
  2. 它的运行结果与先前的类似,但是,它可以获得额外的信息输出至stdout.log文件中。在这我们使用了logger中的info方法。
  3. 消息可以使用各种优先级发送,通过使用sever(),warning(),info(),config(),fine(),finer(),finest()方法。finest()方法用来记录最好的信息,而sever()方法用来记录最严重的信息。使用Log4J 框架来将消息记录在不同的文件中,这些消息基于严重程度和重要性来进行分类。
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@page import="java.util.logging.Logger" %>

<html>
<head><title>Logger.info</title></head>
<body>
<% 
    Logger logger=Logger.getLogger(this.getClass().getName());
%>
<c:forEach var="counter" begin="1" end="10" step="1" >
	<c:set var="myCount" value="${counter-5}" />
	<c:out value="${myCount}"/></br>
<% 
    String message = "counter=" + pageContext.findAttribute("counter") + "myCount=" + pageContext.findAttribute("myCount");
    logger.info( message );   
%>
</c:forEach>
</body>
</html>
  • 调试工具

NetBeans是树形结构,是开源的Java综合开发环境,支持开发独立的Java应用程序和网络应用程序,同时也支持JSP调试。

  • JDB Debugger

  • 客户端和服务器的头模块

有时候,当JSP没有按照预定的方式运行时,查看未加工的HTTP请求和响应也是很有用的。可以直接观察request和response然后看看这些头模块到底怎么了。

  • 小技巧
  1. 使用浏览器显示原始的页面内容,用来区分是否是格式问题。这个选项通常在View菜单下。

  2. 确保浏览器在强制重新载入页面时没有捕获先前的request输出。若使用的是Netscape Navigator浏览器,则用Shift-Reload;若使用的是IE浏览器,则用Shift-Refresh。

JSP国际化

国际化(i18n):表明一个页面根据访问者的语言或国家来呈现不同的翻译版本。

本地化(l10n):向网站添加资源,以使它适应不同的地区和文化。比如网站的印度语版本。

区域:这是一个特定的区域或文化,通常认为是一个语言标志和国家标志通过下划线连接起来。比如"en_US"代表美国英语地区。

JSP容器能够根据request的locale属性来提供正确地页面版本。接下来给出了如何通过request对象来获得Locale对象的语法:

java.util.Locale request.getLocale() 
  • 检测Locale常用方法
方法描述
String getCountry()返回国家/地区码的英文大写,或 ISO 3166 2-letter 格式的区域
String getDisplayCountry()返回要显示给用户的国家名称
String getLanguage()返回语言码的英文小写,或ISO 639 格式的区域
String getDisplayLanguage()返回要给用户看的语言名称
String getISO3Country()返回国家名称的3字母缩写
String getISO3Language()返回语言名称的3字母缩写
<%@ page import="java.io.*,java.util.Locale" %>
<%@ page import="javax.servlet.*,javax.servlet.http.* "%>
<%    
	//获取客户端本地化信息    
	Locale locale = request.getLocale();    
	String language = locale.getLanguage();    
	String country = locale.getCountry(); 
%>
  • 区域日期时间

使用java.text.DateFormat类和它的静态方法getDateTimeInstance()来格式化日期和时间。根据指定的区域来格式化日期和时间:

<%@ page import="java.io.*,java.util.Locale" %>
<%@ page import="javax.servlet.*,javax.servlet.http.* "%>
<%@ page import="java.text.DateFormat,java.util.Date" %>

<%     
	String title = "Locale Specific Dates";     
	//Get the client's Locale     
	Locale locale = request.getLocale( );     
	String date = DateFormat.getDateTimeInstance(DateFormat.FULL, DateFormat.SHORT, locale).format(new Date( )); 
%>
  • 区域货币

使用java.text.NumberFormat类和它的静态方法getCurrencyInstance()来格式化数字。如何根据指定的区域来格式化货币:

<%@ page import="java.io.*,java.util.Locale" %>
<%@ page import="javax.servlet.*,javax.servlet.http.* "%>
<%@ page import="java.text.NumberFormat,java.util.Date" %>

<%     
	String title = "Locale Specific Currency";     
	//Get the client's Locale     
	Locale locale = request.getLocale( );     
	NumberFormat nft = NumberFormat.getCurrencyInstance(locale);     
	String formattedCurr = nft.format(1000000); 
%>
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值