JSP概念

来源:互联网 发布:java的mvc框架详解 编辑:程序博客网 时间:2024/05/21 11:31

http://my.oschina.net/abian/blog/163205

一、jsp基础1.JSP概念Servlet是j2ee提供的动态资源开发技术,是以java的程序的形式进行开发,在java中书写HTML标签是一件十分头疼的事情,所以人们开发出了JSP,看起来像是HTML一样,但是通过服务器的编译最终可以生成Servlet2.JSP的组成部分2.1模版元素直接书写在JSP中的HTML内容,看起来就像写HTML一样的方便,但是最终会在被翻译成Servlet的过程中 out.write()直接输出2.2脚本表达式<%= 表达式%> 接受的是一段java表达式,在JSP翻译到Servlet的过程中,将会计算表达式的值,利用out.write()输出出去2.3脚本片段<%  %>直接可以在脚本片段中书写java源代码,其中的源代码将会直接拷贝到翻译过来的servlet中的响应位置上。2.4JSP声明<%! %>在其中可以写java代码,其中的源代码会被拷贝到servlet中的service方法之外,可以利用它来为servlet增加成员方法、成员变量、静态代码块2.5JSP注释<%-- --%>被jsp注释包围起来的内容将不会被服务器翻译到servlet之中,要注意区分html注释和java注释的区别jsp注释不会被翻译到servlet中,会在翻译时遗弃java注释不会被编译到class文件中,会在编译时遗弃html注释将会当作模版元素,直接输出到浏览器中,浏览器将不会显示html注释中的内容2.6JSP指令2.6.1page指令用来通知翻译引擎,如果翻译当前的JSP[ language="java" ] 当前JSP使用的开发语言[ extends="package.class" ] 当前jsp翻译成servlet后要继承的类,注意此值必须是一个servlet的子类,一般情况下不要改[ import="{package.class | package.*}, ..." ] 导入需要使用到的包 java.lang.*;javax.servlet.*;javax.servlet.jsp.*;javax.servlet.http.*;[ session="true | false" ] 用来指定当前页面是否使用session,如果设置为true,则翻译过来的servlet中将会有对session对象的引用,于是可以直接在jsp中使用session隐式对象。但是这将导致一旦访问jsp就会调用request.getSession()方法,可能导致不必要的空间浪费。如果确定jsp中不需要session可以设为false[ buffer="none | 8kb | sizekb" ] out隐式对象所使用的缓冲区的大小[ autoFlush="true | false" ] out隐式对象是否自动刷新缓冲区,默认为true,不需要更改[ isThreadSafe="true | false" ] 翻译过来的servlet是否实现SingleThreadModel[ errorPage="relative_url" ] 如果页面出错,将要跳转到的页面,除了在jsp中使用此属性指定错误页面外也可以在web.xml中配置整个web应用的错误页面,如果两个都设置则jsp中的此属性起作用[ isErrorPage="true | false" ] 如果设置此属性为true,翻译过来的servlet中将含有Exception隐式对象,其中封装的就是上一个页面中抛出的异常对象[ contentType="mimeType [ ;charset=characterSet ]" | "text/html ; charset=ISO-8859-1" ] 和jsp乱码相关的指令,用来指定jsp输出时,设置的Content-Type响应头用来指定浏览器打开的编码[ pageEncoding="characterSet | ISO-8859-1" ] 服务器翻译jsp时使用的编码集.如果向防止jsp乱码,应该保证文件的保存编码和jsp翻译成servlet用的编码以及输出到浏览器后浏览器打开的编码一致.此属性一旦设置好,翻译引擎会间接帮我们设置content-type属性.[ isELIgnored="true | false" ] 当前页面是否使用el表达式,设置为false时表示启用el,j2ee4.0开始默认支持,j2ee4.0一下做开发时,如果要使用el表达式,需将此属性设置为fals2.6.2include指令<%@ incluede file=""%> 静态引入其他页面的内容*静态引入:在源文件级别进行合并,多个jsp生成一个servlet,最终由这一个servlet生成响应。推荐使用。*动态引入:在运行时将多个输出进行合并,多个jsp分别生成多个servlet,最终由这多个servlet生成响应,组成一个输出流,提供响应。执行效率没有静态引入高。2.6.3taglib指令<%@ taglib uri="" prefix=""%>用来引入标签库。uri指定被引入.tl文件的名称空间prefix 对该名称空间的一个缩写2.7九大隐式对象configapplicationrequestresponsesessionoutpagepageContextException2.7.1out对象可以将他理解成response.getWriter()获得的PrintWriter.它自带一个缓冲区,其大小收page指令中的buffer的设定限制。当缓冲区满或缓冲区被关闭或当前jsp页面结束,则此缓冲区中的内容将被刷新到response.getWriter()的缓冲区中。2.7.2PageContext对象(1)获取其它八大隐式对象,可以认为是一个入口对象。(2)获取其所有域中的数据pageContext操作所有域中属性的方法public java.lang.Object?getAttribute(java.lang.String?name,int?scope)public void setAttribute(java.lang.String?name, java.lang.Object?value,int?scope)public void?removeAttribute(java.lang.String?name,int?scope)pageContext中代表域的常量PageContext.APPLICATION_SCOPEPageContext.SESSION_SCOPEPageContext.REQUEST_SCOPEPageContext.PAGE_SCOPE findAttribute方法,在四大域中搜寻属性,搜寻的顺序是page域、request域、session域、application域,从小域到大域开始搜索,如果搜索到就直接获取该值,如果所有域中都找不到,返回一个null(与el表达式不同,此处返回null,对网页是不友好的)(3)作为域对象使用作用的范围:真个jsp页面,是四大作用域中最小的一个。声明周期:当对jsp的请求开始时生成,当响应结束时销毁。(4)跳转到其他资源其身上提供了forward和include方法,简化重定向和转发的操作========================================================================================================二、标签:JSP既可以用来生成HTML页面,也可以直接书写java源码处理逻辑,这就导致了很多开发者在JSP出现初期,只用JSP做开发,这个JSP页面十分庞大、充满了java源码和HTML标签、许多百分号,逻辑结构混乱,不宜调试程序和页面美化。于是人们希望将java源码从JSP页面中尽量抽离,但是把所有java源码都抽走是不现实的,最基本的获取属性、简单的页面逻辑还是需要的,于是,sun公司就提供了JSP中的标签开发技术,以一个标签代表一种功能的java代码,是整个jsp看起来更像一个HTML,并且不丢失JSP进行逻辑处理的功能。1.JSP标签:由sun公司提供,属于jsp规范中的内容,不需要引入第三方标签库<jsp:inclue>用来替代request.getRequestDispatcher().include()<jsp:forward>用来替代request.getRequestDispatcher().forward()<jsp:param>配合前两个标签使用,可以在包含或转发时,带一些请求参数过去<jsp:useBean id="beanName" class="package.class"  scope="page|request|session|application"/>在指定域中搜寻名字为id设置值的bean,如果没有就在该域中创建一个<jsp:setProperty name="beanName" { property="propertyName" value="{string | <%= expression %>}" |property="propertyName" [ param="parameterName" ] | property= "*" }/> 用于为bean的属性负值name指定bean的名字,property指定要设定的属性的名字,value指定设定的属性的值,param用来指定使用哪个请求参数设定该属性,property可以设置为*,用来将所有的请求参数自动设置懂啊bean对应的属性上<jsp:getProperty name="beanInstanceName" property="PropertyName" />用于获取属性的值输出到输出流中,其中name指定bean的名字,property指定bean上属性的名字2.el表达式:替代<%= %>脚本表达式,在j2ee1.4以前默认是不支持el,如果需要需要指定page指令[isELIgnored="true | false" ]为false,j2ee4.0后默认支持el2.1获得域中的属性${propName}在四个域中搜寻proName属性,输出该值到输出流中${pageScope/requestScope/sessionScope/applicationScope.proName}获取指定域中的属性${attr[0]}获取域中的集合的指定元素${list[0]}获取list中的指定位置元素${map.keyName}获取map中指定键的值${bean.propName}获取javaBean的属性,可以认为是调用了javaBean的getXXX方法,~最重要的一个应用场景:在写路径的时候最好不要把web应用名称写死,java中应该用request.getContextPath去获取。jsp中就用el获取:${pageContext.request.contextPth}~.和[]区别使用点的地方都可以用中括号,如果属性名是数字或包含特殊符号(.-)就必须使用中括号。例子:${map["first.name"]}2.2进行简单运算(1)算数运算:所有参与元算的元素都会被转成数字,如果不能转就报错,空元素参与运算当作没参与。(2)关系运算:(3)逻辑运算:(4)empty/not empty判断一个对象或集合或数组是否为空或长度为0(5)二元表达式 name == null ? "张三" : name;2.3获取web开发常用对象pageContext:代表pageContext对象,注意和pageScope进行区分pageScope:代表page域,可以用来获取page域中的属性reqeustScope:代表reqeust域,可以用来获取reqeust域中的属性sessionScope:代表session域,可以用来获取session域中的属性applicationScope:代表application域,可以用来获取application域中的属性param 代表请求参数组成的map集合${param.userName}paramValues 代表请求参宿组成的map集合,但是此集合的value是String[],用来获取一名多值的paramheader 获取请求头组成的mapheaderValues 获取请求头组成的map但是value是一个String[],用来获取一名多值的headcookie 获取cookie组成的map对象,此map的值是一个cookie对象${cookie.cookieName.cookieValue}initParam 以map封装的web.xml中配置的整个web应用的初始化参数2.4调用java方法el表达式可以调用java中的静态方法,分如下步骤完成:(1)编写一个类,其中应该包含要使用el调用的静态方法(2)编写一个tld文件,描述该方法的调用,在创建tld文件时应选用2.0版本的jsp配置,指定名称空间uri和缩写prefix(3)在tld文件中配置方法信息<function><name>encodeURL</name>el在调用时所使用的方法名<function-class>cn.itheima.util.EncodeURL</function-class>静态方法所在的类全路径名<function-signature>java.lang.String EncodURL( java.lang.String )//对该方法的描述:返回值类型 方法名(参数类型)</function-signature></function>(4)在jsp中使用<%@ taglib uri="" prefix="ppp"%>引入tld文件(5)在jsp中使用${ppp:encodeURL("xxxx")}调用2.5SUN提供的EL函数库,基本上都是对字符串进行操作的函数,请参考张老师写的《第8章 标准标签库_0519.doc》3.JSTL标签库,在javaee4.0需要导入JSTL相关的jar包,在javaee5.0开始,默认已经包含了此jar包。还要需要用<%@ taglib%>指令引入标签库3.1介绍JavaServer Pages Standard Tag Library 由JCP(Java Community Process)指定标准提供给 Java Web 开发人员一个标准通用的标签函数库和 EL 配合来取代传统直接在页面上嵌入 Java 程序(Scripting)的做法,以提高程序可读性、维护性和方便性一般我们使用JSTL1.1以上的版本,应为从这个版本开始支持EL表达式JSTL1.0默认不支持el表达式,不建议使用3.2JSTL标签库****核心标签库 (core)  --- c国际化标签 fmt数据库标签 sql --ServletXML标签  xmlJSTL函数(EL函数) el3.3JSTL核心标签库<c:out> 标签用于输出一段文本内容到pageContext对象当前保存的“out”对象中。<c:set>标签用于把某一个对象存在指定的域范围内,或者设置Web域中的java.util.Map类型的属性对象或JavaBean类型的属性对象的属性。  <c:remove>标签用于删除各种Web域中的属性<c:catch>标签用于捕获嵌套在标签体中的内容抛出的异常,其语法格式如下:<c:catch [var="varName"]>nested actions</c:catch><c:if test=“”>标签可以构造简单的“if-then”结构的条件表达式 <c:choose>标签用于指定多个条件选择的组合边界,它必须与<c:when>和<c:otherwise>标签一起使用。使用<c:choose>,<c:when>和<c:otherwise>三个标签,可以构造类似 “if-else if-else” 的复杂条件判断结构。<c:forEach>标签用于对一个集合对象中的元素进行循环迭代操作,或者按指定的次数重复迭代执行标签体中的内容。 <c:forTokens>用来浏览一字符串中所有的成员,其成员是由定义符号所分隔的<c:param>标签  在JSP页面进行URL的相关操作时,经常要在URL地址后面附加一些参数。<c:param>标签可以嵌套在<c:import>、<c:url>或<c:redirect>标签内,为这些标签所使用的URL地址附加参数。 <c:import> 标签,实现include操作 <c:url>标签用于在JSP页面中构造一个URL地址,其主要目的是实现URL重写。URL重写就是将会话标识号以参数形式附加在URL地址后面 <c:redirect>标签用于实现请求重定向=======================================================================================================================================================三.自定义标签1.自定义标签技术出现的原因:虽然有第三方组织提供了很多标签,但是这些都是一些通用标签,开发中常常需要根据业务需求使用jsp页面,这个时候通用的标签就不够用了,我们需要自己去开发标签库。2.标签处理类生命周期2.1传统标签处理类的生命周期:传统标签处理类在标签被第一次使用时创建实例,从此驻留内存为后续请求提供服务,其中每次标签执行都会依次执行setPageContext(PageContext pc) 、setParent(Tag t) 、doStartTag() 、doEndTag() 、~在服务器停止之前调用release() 2.2简单标签的声明周期:setJspContext(JspContext pc) 、setParent(JspTag parent) 如果没有父标签就不调用、如果有属性就调用对应的setXXX方法如果有el表达式,则计算结果后传入、如果有标签体调用setJspBody(JspFragment jspBody)  、doTag() 执行标签体处理.每次访问该标签时都创建简单标签实例,每次使用后销毁该实例。3.传统标签的继承结构:JspTag(1)传统标签Tag接口-----定义了一个标签处理类应具有的最基本的方法和属性(EVAL_BODY_INCLUDE dostart方法返回表示执行标签体,SKIP_BODY dostart方法用,跳过标签体。EVAL_PAGE 用在doendtag里通知后续页面继续执行,SKIP_PAGE doendtag里通知后续页面不再执行)||----IterationTag接口 (提供了doAfterBody() 在标签体执行过后立即执行,并提供EVAL_BODY_AGAIN/EVAL_BODY_INCLUDE 供doafterbody方法返回表示要重新执行标签体)||----TagSupport类(提供了对pageContext的引用)||--BodyTag接口(EVAL_BODY_BUFFERED 在doStartTag方法中返回通知标签处理器去缓存标签体bodyContent)||----BodyTagSupprot类(getBodyContent() 获取缓存对象bodyContent)(2)简单标签(简单标签的标签体不能包含脚本内容,所以tld文件中配置body-content时不能配置为JSP)SimpleTag接口||---SimpleTagSupport实现类(getJspContext() 获取代表jsp页面的pageContext,getJspBody() 代表标签体内容的JspFragment对象)JspFragment:invoke(Writer out) 此方法一经调用会将标签体输出到对应的流中。如果你直接给一个null默认输出到pageContext.getOut()中如果doTag()方法抛出SkipPageException异常,标签后的jsp页面就不再执行。(3)自定义标签声明属性:在标签处理类中提供setXXX方法,并在tld文件中通过<attribute>标签声明属性的特性即可。4.自定义标签应该具有的最基本的功能实验1:控制jsp页面某一部分内容是否执行。<c:if>实验2:控制整个jsp页面是否执行。实验3:控制jsp页面内容重复执行。<c:forEach>实验4:修改jsp页面内容输出。<c:out> HTML转义实验5:获取属性5.tld文件配置自定义标签<tag><name>demo10</name>-----定义标签的名字<tag-class>cn.itheima.simple.Demo10</tag-class>----标签处理类<body-content>scriptless</body-content>----标签体内容JSP表示任意jsp内容,empty代表只能为空、scriptless带表不能包含脚本内容、tagdependent代表其中的内容是用来给服务器用的,不是输出到浏览器的内容<attribute><name>times</name>---属性的名字(名字必须有)<required>true</required>---是否是一个必须的属性<rtexprvalue>true</rtexprvalue>---是否支持el表达式<type>int</type>---属性的类型</attribute></tag>