javaweb11--jsp

一、jsp基础
1.JSP概念
  Servlet是j2ee提供的动态资源开发技术,是以java的程序的形式进行开发,在java中书写HTML标签是一件十分头疼的事情,所以人们开发出了JSP,看起来像是HTML一样,但是通过服务器的编译最终可以生成Servlet
2.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表达式,需将此属性设置为fals

  2.6.2include指令
   <%@ incluede file=""%> 静态引入其他页面的内容
   *静态引入:在源文件级别进行合并,多个jsp生成一个servlet,最终由这一个servlet生成响应。推荐使用。
   *动态引入:在运行时将多个输出进行合并,多个jsp分别生成多个servlet,最终由这多个servlet生成响应,组成一个输出流,提供响应。执行效率没有静态引入高。
   
  2.6.3taglib指令
   <%@ taglib uri="" prefix=""%>用来引入标签库。
   uri指定被引入.tld文件的名称空间
   prefix 对该名称空间的一个缩写
 
 2.7九大隐式对象
   config
   application
   request
   response
   session
   out
   page
   pageContext
   Exception
  
  2.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_SCOPE
     PageContext.SESSION_SCOPE
     PageContext.REQUEST_SCOPE
     PageContext.PAGE_SCOPE
     
     findAttribute方法,在四大域中搜寻属性,搜寻的顺序是page域、request域、session域、application域,从小域到大域开始搜索,如果搜索到就直接获取该值,如果所有域中都找不到,返回一个null(与el表达式不同,此处返回null,对网页是不友好的)

   (3)作为域对象使用
     作用的范围:整个jsp页面,是四大作用域中最小的一个。
     生命周期:当对jsp的请求开始时生成,当响应结束时销毁。
   
   (4)跳转到其他资源
      其身上提供了forward和include方法,简化重定向和转发的操作

====!!!!!四大作用域总结:==============================================
   ServletContext   application域
    作用范围:整个web应用
    生命周期:当web应用第一次被访问时,服务器创建ServletContext对象,以后就一直存在与服务器内存中,当服务器正常关闭时,该对象会调用destory方法,做好善后工作,随后消亡,或者web应用被移除服务器,该对象也会消亡
   Session          Session域
    作用范围:整个会话范围
    生命周期:第一次明确调用request.getSession()时,被服务器创建,存在内存中,当服务器非正常关闭,或超出存在的时间,一般为30分钟,或调用invalidate()时,Session对象都会被销毁
   Request          Request域
    作用范围:整个请求链
    生命周期:每次请求发出时都会创建ServletRequest 域对象,当这次请求结束,被销毁
   PageContext      page域
    作用范围: 整个jsp页面
    生命周期: 页面被访问是创建,页面访问结束被销毁
   
   如果有一个数据只在jsp页面中使用,存入page域
   如果有一个数据,在请求转发时要带给被转发的资源,存入request域
   如果有一个数据,现在我要用,过一会我还要用,存入session域
   如果有一个数据,除了我现在要用,过一会其他人也要用,存入application域
  ======================================================================
 
 6.jsp杂项知识
  (1)映射jsp
   <servlet>
    <servlet-name>5JSP</servlet-name>
    <jsp-file>/5.jsp</jsp-file>
   </servlet>
   <servlet-mapping>
    <servlet-name>5JSP</servlet-name>
    <url-pattern>/xxx/5jsp</url-pattern>
   </servlet-mapping>

=========================================================================================
二、标签: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后默认支持el
  2.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[],用来获取一名多值的param
   header 获取请求头组成的map
   headerValues 获取请求头组成的map但是value是一个String[],用来获取一名多值的head
   cookie 获取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.spadeking.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 --Servlet
   XML标签  xml
   JSTL函数(EL函数) el
  3.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>标签用于实现请求重定向

 4.自定义标签

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 供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>

一.请求重定向:
 1.原始方式
  response.setStatus(302);
  response.setHeader("Locaton","/xxxxxx");
 2.快捷方式
  response.sendRedirect("/xxxx");
 *3.额外的方式
  response.setHeader("refresh","0;url=/xxxx");
  
  <c:redirect>
二、请求转发
 1.
  ServletContext.getRequestDispatcher().forward();
 2.
  request.getRequestDispatcher().forward();
 3.
  pageContext.forward("");

  <jsp:forward>

三、请求包含
 1.
  ServletContext.getRequestDispatcher().include()
 2.
  request.getRequestDispatcher().include()
 3.
  <%@ include file=""%>
 4.
    pageContext.include("/index.jsp");
   5.
    <jsp:include/>
   6.
    <c:import/>

 

一、为什么:要分层
 使软件具有结构性,便于开发、维护和管理。
 将不同功能模块独立,在需要替换某一模块时不需要改动其他模块,方便代码的复用、替换
二、层与层耦合的概念,利用工厂类解耦
 在分层结构中,我们希望将各个功能
 约束在各自的模块(层)当中的,而当属于某一层的对象、方法“入侵”到了其他层,如将web层的ServletContext对象传入service层,或service层调用XMLDao独有的方法,就会导致层与层之间的关系过于“紧密”,当需要替换某一层时不可避免的要修改其他关联的层,这和我们软件分层最初的设想-----层与层分离,一个层尽量不依赖其他层存在,当修改一层时无需修改另一层的设想是违背的。这种“入侵”造成的“紧密”关系就早做层与层之间发生的“耦合”,而去掉这种耦合性的过程就叫做层与层之间“解耦”
 利用工厂类可以实现解耦的功能
三、如何判断一项功能到底属于哪一层
 某一项功能属于哪一层,往往是不能明确确定出来的,这时可以参考如下标准进行判断:
  此项功能在业务逻辑上更贴近与哪一层,
  放在哪一层更能较少耦合 此项功能是否必须使用某一层特有的对象
  如果放在哪一层都可以,那么放在哪一层更方便技术上的实现,及方便代码的编写和维护
 
四、异常的处理
 如果一个异常抛给上一层会增加程序的耦合性,请当场解决:如将xml解析错误抛给service层,那么当换成mysqldao时,还需要修改service去掉xml解析异常的处理
 如果上一层明确需要此异常进行代码的流转,请抛出:如当查找一个用户信息而用户找不到时,可以抛出一个用户找不到异常,明确要求上一层处理
 如果这一层和上一层都能解决尽量在这一层解决掉
 如果这一层不能解决,而上一层能解决抛给上一层
 如果所有层都不能解决,则应抛出给虚拟机使线程停止,但是如果直接抛出这个异常,则还需要调用者一级一级继续往上抛出最后才能抛给虚拟机,所以还不如在出现异常的位置直接trycatch住后转换为RuntimeException抛出。:如读取配置文件出错,任何层都不能解决,转为RuntimeException抛出,停止线程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值