l       JSP技术

JSP全称是Java Server Pages,它和Servlet技术一样,都是SUN公司定义的一种用于开发动态web资源的技术。

1. jsp 是一种动态web资源的开发技术

jsp的特点:

1) 写jsp就像在写html 所以做数据美化很方便, jsp的特点就是能写java代码

2) jsp中不适合写复杂的java代码

3)  在jsp页面中禁止写java代码

4)  在jsp页面中避免不了要写java代码,

jstl自定义标签、el表达式两个技术就是用于移除jsp页面中的java代码

servlet 和 jsp 都用在哪

Servlet适合写大量的java代码,所以用于处理用户的请求,进行逻辑判断

jsp负责做数据的输出,因为它可以轻松地对数据进行美化

为什么JSP也是一种动态web资源的开发技术呢?

写JSP虽然就像是在写html,但Jsp技术允许在页面中嵌套java代码,并且允许开发人员在页面中获取request、response等web开发常用对象,实现与浏览器的交互,所以jsp也是一种动态web资源的开发技术。

Jsp快速入门:在jsp页面中输出当前时间。

<body>

    <%= new Date().toLocaleString() %>

</body>

l       不管是JSP还是Servlet,虽然都可以用于开发动态web资源。但由于这2门技术各自的特点,在长期的软件实践中,人们逐渐把servlet作为web应用中的控制器组件来使用,而把JSP技术作为数据显示模板来使用。

l       其原因为,程序的数据通常要美化后再输出:

  • 让jsp既用java代码产生动态数据,又做美化会导致页面难以维护。
  • 让servlet既产生数据,又在里面嵌套html代码美化数据,同  样也会导致程序可读性差,难以维护。
  • 因此最好的办法就是根据这两门技术的特点,让它们各自负责各的,servlet只负责响应请求产生数据,并把数据通过转发技术带给jsp,数据的显示jsp来做。

l       JSP脚本表达式

JSP脚本表达式(expression)用于将程序数据输出到客户端

       语法:<%= 变量或表达式 %>

       举例:当前时间:<%= new java.util.Date() %>

JSP引擎在翻译脚本表达式时,会将程序数据转成字符串,然后在相应位置用out.print(…) 将数据输给客户端。

JSP脚本表达式中的变量或表达式后面不能有分号(;)。

l       JSP脚本片断

JSP脚本片断(scriptlet)用于在JSP页面中编写多行Java代码。语法:

<%

              多行java代码

%>

注意:JSP脚本片断中只能出现java代码,不能出现其它模板元素, JSP引擎在翻译JSP页面中,会将JSP脚本片断中的Java代码将被原封不动地放到Servlet的_jspService方法中。

JSP脚本片断中的Java代码必须严格遵循Java语法,例如,每执行语句后面必须用分号(;)结束。

在一个JSP页面中可以有多个脚本片断,在两个或多个脚本片断之间可以嵌入文本、HTML标记和其他JSP元素。

       举例:

<%

       int x = 10;

       out.println(x);

%>

<p>这是JSP页面文本</p>

<%

       int y = 20;

       out.println(x);

       %>

多个脚本片断中的代码可以相互访问,犹如将所有的代码放在一对<%%>之中的情况。如:out.println(x);

l       单个脚本片断中的Java语句可以是不完整的,但是,多个脚本片断组合后的结果必须是完整的Java语句,例如:

<%

       for (int i=1; i<5; i++)

       {

%>

       <H1>www.it315.org</H1>

<%

       }

%>

l       JSP声明

JSP页面中编写的所有代码,默认会翻译到servlet的service方法中, 而Jsp声明中的java代码被翻译到_jspService方法的外面。语法:

<%!

       java代码

%>

所以,JSP声明可用于定义JSP页面转换成的Servlet程序的静态代码块、成员变量和方法 。

多个静态代码块、变量和函数可以定义在一个JSP声明中,也可以分别单独定义在多个JSP声明中。

JSP隐式对象的作用范围仅限于Servlet的_jspService方法,所以在JSP声明中不能使用这些隐式对象。

l       JSP注释

JSP注释的格式:

              <%-- 注释信息 --%>

JSP引擎在将JSP页面翻译成Servlet程序时,忽略JSP页面中被注释的内容。

l       JSP指令

l       JSP指令(directive)是为JSP引擎而设计的,它们并不直接产生任何可见输出,而只是告诉引擎如何处理JSP页面中的其余部分。在JSP 2.0规范中共定义了三个指令:

l       page指令

l       Include指令

l       taglib指令

l       JSP指令的基本语法格式:

       <%@ 指令 属性名="值" %>

       举例:<%@ page contentType="text/html;charset=gb2312"%>

l       如果一个指令有多个属性,这多个属性可以写在一个指令中,也可以分开写。

       例如:

              <%@ page contentType="text/html;charset=gb2312"%>

              <%@ page import="java.util.Date"%>

       也可以写作:

       <%@ page contentType="text/html;charset=gb2312" import="java.util.Date"%>

l       Page指令

1. page指令

import 导包 多个包用逗号隔开

session  指定是否使用session

errorPage 指定出错跳转的页面

也可以在web.xml文件中进行配置

<error-page>

              <exception-type>java.lang.ArithmeticException</exception-type>

              <location>/error.jsp</location>

       </error-page>

      

       <error-page>

              <error-code>404</error-code>

              <location>/404.jsp</location>

       </error-page>

       <error-page>

              <error-code>500</error-code>

              <location>/500.jsp</location>

       </error-page>

l        

l       page指令用于定义JSP页面的各种属性,无论page指令出现在JSP页面中的什么地方,它作用的都是整个JSP页面,为了保持程序的可读性和遵循良好的编程习惯,page指令最好是放在整个JSP页面的起始位置。

l       JSP 2.0规范中定义的page指令的完整语法:

<%@ page

       [ language="java" ]

       [ extends="package.class" ]

       [ import="{package.class | package.*}, ..." ]

JSP引擎自动导入下面的包 lang servlet.* servlet.jsp.* servlet.http.*

可以在一条page指令的import属性中引入多个类或者包 其中每个包或类之间用逗号分隔

<%@ page import =“ java.util.Date, java.sql.*,java.io.*” %>

上面的语句可以改写为多条page指令的import属性来分别引入哥哥包或者类

<%@ page import = “java.util.Date ”%>

<%@ page import = “java.sql.* ”%>

<%@ page import = “java.io.* ”%>

       [ session="true | false" ]

       [ buffer="none | 8kb | sizekb" ]

       [ autoFlush="true | false" ]

       [ isThreadSafe="true | false" ]

       [ info="text" ]

       [ errorPage="relative_url" ]

errorPage属性的设置必须使用相对路径,如果以“/”开头 表示相对于当前WEB应用程序的根目录 (注意不是站点根目录),否则,表示相对于当前页面

可以在web.xml文件中使用<error-page>元素为整个WEB应用程序设置错误处理页面 其中的<exception-type>子元素指定异常类的完全限定名,<location>元素指定以“/”开头的错误处理页面的路径。

如果设置了某个JSP页面的errorpage属性 那么在web.xml文件中设置的错误处理将不对该页面起作用 。

       [ isErrorPage="true | false" ]

       [ contentType="mimeType [ ;charset=characterSet ]" | "text/html ; charset=ISO-8859-1" ]

JSP引擎(服务器)会根据page指令的contentType属性生成相应的调用servletresponse setcontenttype方法的语句

Page指令的contenttype属性还具有说明JSP源文件的字符编码的作用

       [ pageEncoding="characterSet | ISO-8859-1" ]

       [ isELIgnored="true | false" ]

%>

l       使用page指令解决JSP中文乱码

 a.jsp

中国

gb2312

98 99

jsp引擎

98 99  iso-8859-1

         %$$#

Servlet

setContentType(..gb2312);

  %$$#

Servlet引擎

%$$#

response

 iso-8859-1

 98 99

gb2312  63 63

IE

98 99

iso-8859-1

%$$#

1. 原因: 计算中存储的和传输的都是二进制

                我们需要阅读文字,就需要用到码表, 字符—> 字节 为编码过程  字节->字符 解码过程

                     编码和解码使用了不同码表就会出现乱码

2. 乱码问题一般出现在解错码的情况

       1) 我们的程序接错码, 需要知道错误的码表,编码回去,再用正确的码表解码

       2) 别的程序解析我们的文件解错码, 需要告诉对方我们使用码表

JSP程序存在有与Servlet程序完全相同的中文乱码问题

输出响应正文时出现的中文乱码问题

读取浏览器传递的参数信息时出现的中文乱码问题

JSP引擎将JSP页面翻译成Servlet源文件时也可能导致中文乱码问题

JSP引擎将JSP源文件翻译成的Servlet源文件默认采用UTF-8编码,而JSP开发人员可以采用各种字符集编码来编写JSP源文件,因此,JSP引擎将JSP源文件翻译成Servlet源文件时,需要进行字符编码转换。

如果JSP文件中没有说明它采用的字符集编码,JSP引擎将把它当作默认的ISO8859-1字符集编码处理。

如何解决JSP引擎翻译JSP页面时的中文乱码问题

通过page指令的contentType属性说明JSP源文件的字符集编码

page指令的pageEncoding属性说明JSP源文件的字符集编码

l     include指令

l       include指令用于引入其它JSP页面,如果使用include指令引入了其它JSP页面,那么JSP引擎将把这两个JSP翻译成一个servlet。所以include指令引入通常也称之为静态引入。

l       语法:

       <%@ include file="relativeURL"%>

       其中的file属性用于指定被引入文件的路径。路径以“/”开头,表示代表当前web应用。

l       细节:

ü       被引入的文件必须遵循JSP语法。

ü       被引入的文件可以使用任意的扩展名,即使其扩展名是html,JSP引擎也会按照处理jsp页面的方式处理它里面的内容,为了见明知意,JSP规范建议使用.jspf(JSP fragments)作为静态引入文件的扩展名。

ü       由于使用include指令将会涉及到2个JSP页面,并会把2个JSP翻译成一个servlet,所以这2个JSP页面的指令不能冲突(除了pageEncoding和导包除外)。

ü        

l     JSP运行原理和九大隐式对象

l       每个JSP 页面在第一次被访问时,WEB容器都会把请求交给JSP引擎(即一个Java程序)去处理。JSP引擎先将JSP翻译成一个_jspServlet(实质上也是一个servlet) ,然后按照servlet的调用方式进行调用。

l       由于JSP第一次访问时会翻译成servlet,所以第一次访问通常会比较慢,但第二次访问,JSP引擎如果发现JSP没有变化,就不再翻译,而是直接调用,所以程序的执行效率不会受到影响。

l       JSP引擎在调用JSP对应的_jspServlet时,会传递或创建9个与web开发相关的对象供_jspServlet使用。JSP技术的设计者为便于开发人员在编写JSP页面时获得这些web对象的引用,特意定义了9个相应的变量,开发人员在JSP页面中通过这些变量就可以快速获得这9大对象的引用。

l       这9个对象分别是哪些,以及作用也是笔试经常考察的知识点。

ServletContext   application              Servlet的上下文路径

ServletConfig     config

HttpServletResponse  response

HttpServletRequest  request

HttpSession           session           jsp配置了 page指令 session=true(默认)  的情况下才有

this                     page

Throwable             exception        jsp配置了 isErrorPage=”true”的情况下才有

JspWriter               out                        向ie输出数据  response.getWriter 获得一个PrintWriter              out 就是 PrintWriter的缓冲流

PageContext          pageContext    jsp页面的上下文

1. out 对象

2. pageContext对象

可以作为域对象,被称作page域

有个方法是 findAttribute() 可以从四个域中根据指定的key去取值,从范围小的开始找

pageContext.getRequest().getAttribute();

最重要的作用就是可以获得其他8个隐式对象

标签和el表达式都是用于移除jsp中的java代码

所有的java代码都可以移到另一个java类的方法中,然后再复用代码

但是,如果用到了九大隐式对象,这样的java代码就不容易移到方法中,为了能移到方法中需要将

九个对象传给方法

一个比较方便的做法就是将pageContext对象传递给后台的方法,在方法中可以轻松地获得其他8个对象

l     out隐式对象

l       out隐式对象用于向客户端发送文本数据。

l       out对象是通过调用pageContext对象的getOut方法返回的,其作用和用法与ServletResponse.getWriter方法返回的PrintWriter对象非常相似。

l       JSP页面中的out隐式对象的类型为JspWriter,JspWriter相当于一种带缓存功能的PrintWriter,设置JSP页面的page指令的buffer属性可以调整它的缓存大小,甚至关闭它的缓存。

l       只有向out对象中写入了内容,且满足如下任何一个条件时,out对象才去调用ServletResponse.getWriter方法,并通过该方法返回的PrintWriter对象将out对象的缓冲区中的内容真正写入到Servlet引擎提供的缓冲区中:

ü       设置page指令的buffer属性关闭了out对象的缓存功能

ü       out对象的缓冲区已满

ü       整个JSP页面结束

l       out隐式对象的工作原理图

l       pageContext对象

l       pageContext对象是JSP技术中最重要的一个对象,它代表JSP页面的运行环境.

l       这个对象不仅封装了对其它8大隐式对象的引用,

l       它自身还是一个域对象,可以用来保存数据。

l       并且,这个对象还封装了web开发中经常涉及到的一些常用操作,例如引入和跳转其它资源、检索其它域对象中的属性等。

l     通过pageContext获得其他对象

l       getException方法返回exception隐式对象

l       getPage方法返回page隐式对象

l       getRequest方法返回request隐式对象

l       getResponse方法返回response隐式对象

l       getServletConfig方法返回config隐式对象

l       getServletContext方法返回application隐式对象

l       getSession方法返回session隐式对象

l       getOut方法返回out隐式对象

l       pageContext封装其它8大内置对象的意义,思考:如果在编程过程中,把pageContext对象传递给一个普通java对象,那么这个java对象将具有什么功能? 

l     pageContext作为域对象

l       pageContext对象的方法

l       public void setAttribute(java.lang.String name,java.lang.Object value)

l       public java.lang.Object getAttribute(java.lang.String name)

l       public void removeAttribute(java.lang.String name)

l       pageContext对象中还封装了访问其它域的方法

l       public java.lang.Object getAttribute(java.lang.String name,int scope)

l       public void setAttribute(java.lang.String name, java.lang.Object value,int scope)

l       public void removeAttribute(java.lang.String name,int scope)

l       代表各个域的常量

l       PageContext.APPLICATION_SCOPE

l       PageContext.SESSION_SCOPE

l       PageContext.REQUEST_SCOPE

l       PageContext.PAGE_SCOPE

l       findAttribute方法    (*重点,查找各个域中的属性)

l     引入和跳转到其他资源

l       PageContext类中定义了一个forward方法和两个include方法来分别简化和替代RequestDispatcher.forward方法和include方法。

l       方法接收的资源如果以“/”开头, “/”代表当前web应用。

web开发的四个域对象

pageContext(称之为page域) PageContext

request     (称之为request域)HttpServletRequest

session     (称之为session域)HttpSession

application(称之为application域)ServletContext

这4个对象的生命周期?

page:      jsp页面被执行到执行完毕

request:   浏览器发送请求到当次请求结束

                        浏览器发送请求,web容器创建request对象,请求结束,该对象销毁

session:  开始一个会话,服务器创建session,一直到session失效

application: web应用发布到服务器,服务器启动,对象就创建,直到web应用被移除或服务器关闭该对象被销毁

什么是域?为什么把这4个对象叫做域对象呢?

域即范围, 这4个对内部都维护了map集合,可以存储数据,这4个对象都有固定的生命周期和作用范围,所以被称作域对象

四个域的范围(作用域)

page:  只在当前jsp页面有效

request:  只在一次请求中有效

session:  只在当前会话有效

application: 整个web应用都有效

哪种情况下用哪种域对象。

page:    想在页面用map就用page

request: 需要显示数据给浏览器看完了就没用了,就存在request域 (典型应用请求转发)

session: 数据给用户看完了,一会还有用,就存在session域

                     比如,用户登陆,购物车

                     典型应用: 请求重定向

application: 数据给用户用完了,别的用户也要用

                     比如,聊天信息、统计在线人数

四个域的范围是从小到大的: page、request、session、application

使用的时候有个原则: 小的域对象能搞定的绝对不用大的

七、javabean

1. 一个java类

特点:

必须有无参的构造函数  通过反射的方式来访问 调用class的newinstance它是调用匿名的构造器创建对象 如果没有创建不了

属性私有(字段)

对外要暴露共有的 getter 或 setter

2. 用反射的方法访问javabean为内省

3. BeanUtils应用

请求参数封装到javabean的通用方法

public class WebUtils {

 

       private WebUtils(){}

      

       // 通用方法 将请求参数封装到bean

       public static <T> T request2Bean(HttpServletRequest request, Class<T> clazz) {

              try {

                     // 创建 javabean

                     T bean = clazz.newInstance();

                     // 注册一个转换器

                     ConvertUtils.register(new DateLocaleConverter(), Date.class);

                     // 获得所有的参数名称

                     Enumeration e = request.getParameterNames();

                     while (e.hasMoreElements()) {

                            String name = (String) e.nextElement();

                            String value = request.getParameter(name);

                            BeanUtils.setProperty(bean, name, value);

                     }

                     return bean;

              } catch (Exception e) {

                     throw new RuntimeException(e);

              }

       }

}

工具类: 构造函数私有化 对外提供静态的公有的方法  凡是用到传参数的时候都可以用到这个工具类

posted on 2011-05-03 22:01  情定诺坎普  阅读(706)  评论(1编辑  收藏  举报