JSP request.setAttribute()详解及实例
这篇文章主要介绍了 javascript request.setAttribute()详解及实例的相关资料,需要的朋友可以参考下
javascript request.setAttribute()详解
request.setAttribute()怎么用的?
JSP1代码
JSP2代码
为什么JS P2中会找不到test?
用来在同一个request周期中保存变量使用。比如servlet调用后,推出JSP页面,这是一个request周期,如果在Jsp页面需要servlet中的一些 处理结构,就从request.getAttribute中获取。
sendRedirect()方法是通过浏览器重定向的,所以第二个JSP页面中获得的request并非是前一个页面的request(两次请求生成了前后两个不同的 request对象了)。
用RequestDispatcher接口的forward()方法试试看。
都在一起使用
可得到JSP页面一表单中控件的Value。其实表单控件中的Object的 name与value是存放在一个哈希表中的,所以在这里给出Object的name会到哈希表中找出对应它的value。
而不同页面间传值使用request.setAttribute(position, nameOfObj)时,只会从a.jsp到b.jsp一次传递,之后这个request就会失去它的作用范围,再传就要再设一个 request.setAttribute()。而使用session.setAttribute()会在一个过程中始终保有这个值。
P.S:JavaScript与JSP中不能相互传值,因为JavaScript运行在客户端,而JSP运行在服务器端。若想使它们之间可以相互传递参数,可以在JSP中设置一个hidden控件,用它的value结合上面所说的用法来传递所需的数值。
- request.getRequestDispatcher()是请求转发,前后页面共享一个request
- response.sendRedirect()是重新定向 前后页面不是一个request
- sendRedirect是通知IE再次提交一个http请求, dispatcher是直接转到目的地址,最直接得表现,应该是dispatcher后的IE地址栏地址是不变的
- request.getRequestDispatcher()是服务器端跳转
- response.sendRedirect()是客户端跳转
关于request.request.getRequestDispatcher().forward()方法的一点看法
今天在调试一个Servlet程序的时候,报了以下错误!
根据字面理解的话,意识是在response已经提交后程序不能再一次的跳转!
研究代码以后发现,是因为前面已经执行过一次request.request.getRequestDispatcher().forward()
但是后面的request.request.getRequestDispatcher().forward()依然被执行到了!
那么很有理由相信request.request.getRequestDispatcher().forward()跳转本身是不会返回什么,也不会终止程序体的执行!
程序体后面该执行的还是会执行!
sendRedirect()是请求从定向,和超连接是一个意思,比如你在A页面中写一个request.setAtribute,sendRedirect到B页面,就是说服务器从A页面中给你一个response,然后你的浏览器再去request到B页面,由于有两次request和response,是不能在B页面取到request.setAtribute里的值,能从地址栏看到url的改变
request.getRequestDispatcher().forward(request,response)是请求分发器,比如你在A页面中写一个request.setAtribute,request.getRequestDispatcher().forward(request,response)到B页面,那就是说服务器给你的response是B页面的,并且只有一次request和response,所以是能在B页面取到request.setAtribute里的值,地址栏的url仍然是A页面的
request.sendRedirect 与 request.getRequestDispatcher.forward 的区别
1.forward重定向是在容器内部实现的同一个Web应用程序的重定向,所以forward方法只能重定向到同一个Web应用程序中的一个资源,重定向后浏览器地址栏URL不变,而sendRedirect方法可以重定向到任何URL, 因为这种方法是修改http头来实现的,URL没什么限制,重定向后浏览器地址栏URL改变。
2.forward重定向将原始的HTTP请求对象(request)从一个servlet实例传递到另一个实例,而采用sendRedirect方式两者不是同一个application。即参数的传递方式不一样:forward的form参数跟着传递,所以在第二个实例中可以取得HTTP请求的参数。sendRedirect只能通过链接传递参数,response.sendRedirect(“login.jsp?param1=a”)。
3.sendRedirect能够处理相对URL,自动把它们转换成绝对URL,如果地址是相对的,没有一个‘/\’,那么Web container就认为它是相对于当前的请求URI的。比如,如果为response.sendRedirect(\”login.jsp\”),则会从当前servlet 的URL路径下找login.jsp: http://localhost:8081/dms/servlet/Servlet 重定向的URL: http://localhost:8081/dms/servlet/login.jsp,如果为response.sendRedirect(\”/login.jsp\”)则会从当前应用径下查找url:http://localhost:8081/login.jsp。而forward不能这样处理相对路径。
感谢阅读!
JSP运行原理及运行过程
JSP的工作模式是请求/响应模式,客户端首先发出HTTP请求,JSP程序收到请求后将进行处理并返回处理结果。在一个JSP文件第一次被请求的时候,JSP引擎(容器)把该JSP文件转换成一个Servlet,而这个引擎本身也是一个Servlet。JSP的运行原理如图11-4所示。
图11-4 JSP的运行原理
JSP的运行过程具体如下。
(1)客户端发出请求,请求访问JSP文件。
(2)JSP容器先将JSP文件转换成一个Java源文件(Java Servlet源程序),在转换过程中,如果发现JSP文件中存在任何语法错误,则中断转换过程,并向服务端和客户端返回出错信息。
(3)如果转换成功,则JSP容器会将生成的Java源文件编译成相应的字节码文件*.class。该class文件就是一个Servlet,Servlet容器会像处理其他Servlet一样处理它。
(4)由Servlet容器加载转换后的Servlet类(class文件)创建一个该Servlet(JSP页面的转换结果)的实例,并执行Servlet的jspInit()方法。jsInit()方法在Servlet的整个生命周期中只会执行一次。
(5)执行jspService()方法处理客户端的请求。对于每一个请求,JSP容器都会创建一个新的线程处理它。如果多个客户端同时请求该JSP文件,则JSP容器会创建多个线程,使每一个客户端请求都对应一个线程。
(6)如果JSP文件被修改了,则服务器将根据设置决定是否对该文件重新进行编译,如果需要重新编译,则使用重新编译后的结果取代内存中的Servlet,并继续上述处理过程。需要注意的是,虽然JSP效率很高,但在第一次调用时往往由于需要转换和编译,所以会产生一些轻微的延迟。
(7)如果系统出现资源不足等问题,JSP容器可能会以某种不确定的方式将Servlet从内存中移除,发生这种情况的时候,首先会调用jspDestroy()方法,然后Servlet实例会被作为“垃圾”进行处理。
(8)当请求处理完成后,响应对象由JSP容器接收,并将HTML格式的响应信息发送回客户端。
深入理解JSP
在前面的登录案例中,登录失败后为了能够响应登录的错误信息。我特意创建了一个LoginErrorServlet用来动态地拼接错误信息。
【代码如下】
- LoginErrorServlet:登录失败,手动拼接login.html页面,并将错误信息动态地添加进去。
2. LoginErrorServlet:手动拼接html页面,动态展示登录错误信息。
说明:上述虽然能够达到我的需求。但是他有以下两个缺点:
- 拼接麻烦开发效率低;
- 阅读性差,代码难以维护;
问题:思考既然以前在servlet有上述缺点,为什么还要使用拼接呢?
因为如果我们想在html页面显示错误的信息,我们只能采用这种方式,html页面不能够书写java代码。而我们既不采用上述拼接标签的方式,还想实现简单的效果。我们可以使用jsp技术,在jsp中可以书写java代码同时还可以书写html标签。这样 就能够使用jsp技术在客户端和服务端直接进行数据交互了。
JSP全名为Java Server Pages,中文名叫java服务器页面,其本质是一个Servlet ,它是由Sun 公司倡导、许多公司参与一起建立的一种动态网页 技术标准。其实就是一个既能书写Servlet代码又能书写HTML代码的文件。
【图1 创建JSP】
文件内容如下:
jsp中既可以书写java代码,同时还可以书写html标签。我们完成练习如何在jsp页面中书写java代码。
【需求】
- 在jsp页面上获取当前日期,并将其格式化成字符串\”yyyy-MM-dd HH:mm:ss\”;
- 将这个字符串展示在浏览器上;
说明:
1)jsp中的注释:<%–注释内容–%>,添加注释的快捷键是:ctrl+/;
2)在jsp中书写代码有三种方式,我们这里先介绍一种,格式:
<%java代码%>
【参考代码】demo01.jsp
【运行结果】启动tomcat,并且访问这个jsp页面。
在浏览器看到 format格式的日期:
【控制台输出】
通过上面的jsp初体验, 我们看到jsp中既可以编写java代码也可以直接编写html代码,相对servlet更加方便.并且我们通过浏览器访问到的jsp页面,最后在浏览器中看到了响应到浏览器中的结果。那么整个过程的执行流程是怎样的呢,我们可以参考如下图:
通过上面的执行流程,我们可以找到jsp生成的.java源文件,这样就可以知道jsp的真正原理,同时可以看下jsp为什么是servlet。
如何查看生成的.java源文件呢?
我们可以到tomcat中查看一下上面案例中的jsp页面是怎么样的一种存在。IDEA借助tomcat发布web项目的机制:动态发布(为每一个web项目创建一个单独的发布文件)。我们可以通过tomcat其中日志中的CATALINA_BASE路径找到我们当前这个JSP页面在tomcat中的位置:
【图1 IDEA发布项目的路径】
找到这个目录后,会看到以下3个文件夹:
【图2 文件目录】
我们打开C:\\Users\\admin.IntelliJIdea2017.3\\system\\tomcat\\Tomcat_8_5_311_day05目录发现有两个文件:
【图3 JSP被翻译后的代码】
打开demo01_jsp.java文件后,代码如下所示:
我们可以看到当前的jsp文件被翻译成了一个类,这个类继承HttpJspBase类,那么这个HttpJspBase类又是什么?
注意:jsp的翻译由服务器完成,HttpJspBase类一定也是tomcat服务器的内容,所以我们需要找到org.apache.jasper.runtime.HttpJspBase这个类,这个类可以到tomcat的安装目录下的lib文件夹下找到jasper.jar,将其解压,找如下目录:org\\apache\\jasper\\runtime,找到HttpJspBase.class。
使用反编译工具打开,如下所示:
通过观察源码,根据我们所学习的继承关系,我们发现JSP其实底层就是一个servlet。通过观察源码,我们发现我们刚刚编写的所有代码都在该Servlet里面的service方法内部。
总结:
1.<% %>中书写的代码被直接解析成java代码;
2.jsp之所以可以编写html代码,其实本质上也是类似我们使用Servlet直接输出的。也就是说底层还是像我们之前做法一样进行标签拼接。html部分都被out.write(\”\”)方法以字符串的形式拼接,然后响应给浏览器;
3.在这个java文件中有个_jspService,这个方法有两个参数request,response;
4.由于我们自己编写的代码全部都落入到了service方法内部一开始就已经声明了request,response,session,application(ServletContext)等对象了,这些对象成为之jsp内置对象。
在JSP页面的body标签中,可以直接书写html代码和JS代码。但是,如果在JSP页面中书写java代码。必须遵循固定的格式,才能够生效;JSP页面中书写java代码有三种方式:1.脚本片段;2.脚本声明;3.脚本表达式;
脚本片段指的是一段java代码。书写格式:<% java 代码 %>
【示例】 demo02.jsp
【被翻译后的源码】
【页面结果】:
说明:通过查看jsp的源代码我们发现,脚本片段的代码都会存在service方法中,而方法中是不可以定义一个方法的。
脚本声明的格式:<%! 书写Java代码 %>
【示例一】声明成员变量 demo03.jsp
【被翻译后的代码】成员变量
虽然脚本声明和脚本片段已经可以书写Java代码了,但是如果我们要使用java代码向页面输出一些内容,还是需要 使用原来的response对象,比较繁琐,因此,我们需要一个更加简便的方式,可以代替response向页面输出内容 ——这个就是脚本表达式。
脚本表达式的格式:<%= 向浏览器输出的内容 %> 等价于:out.print( 向浏览器输出的内容)
【示例】在页面输出信息
注意:
上面代码中,我们书写代码的顺序是先:
再:
可是浏览器查看数据的结果是
原因:response.getWriter()获取的输出流是:PrintWriter。而脚本表达式<%=%> 等价于:out.print( 向浏览器输出的内容).并且这种方式获取的输出流是:JspWriter。其实导致上述输出结果前后顺序就是PrintWriter和JspWriter。
JspWriter:将输出的内容先放到jsp内置的缓冲区中,然后再刷新输出。
PrintWriter:是不使用jsp内置的缓冲区,直接将内容写到网页中。
总结:
1.不要同时使用脚本表达式和response同时往页面输出信息,会出现顺序不一致。现象:response会出现在最前面。使用一种输出即可。
2.脚本表达式<%= str %> :在页面输出内容,在service方法中起作用;
3.脚本片段<% %> :在service方法中,原样输出的代码片段;属于局部为,放在_jspService方法中
4.脚本声明:<%! String str = \”程序员\” %> :定义成员变量;
JSP指令(directive)是为JSP引擎而设计的,它们并不直接产生任何可见输出效果,而只是告诉引擎如何处理JSP页面中的其余部分。
指令用来声明JSP页面的一些属性,比如编码方式,文档类型。我们在servlet中也会声明我们使用的编码方式和响应的文档类型的,而JSP就是用指令来声明的。
【作用】
【格式】
【作用】
【格式】
【使用示例】
需求:课后的原型中的页面放到web文件夹下面。然后打开主页index.jsp.在主页中引入header.jsp页面。
【效果】
三大指令是什么?分别的作用是?
- page:设置网页上一些属性
- taglib:导入标签库
- include:包含另一个JSP页面
page是jsp中必须使用的一个指令,用于设置JSP上各种页面的属性,告诉tomcat如何将一个JSP翻译成Servlet
可以放在JSP中任何位置,一般建议放在页面的最顶部
方式一: 每个page指令导入一个类
方式二: 一个page指令的import属性导入所有的包,包之间使用逗号分隔
作用:相当于导包
【代码演示】
【作用】设置响应编码
1、通过page指令的errorPage属性指定:如果页面出错,转发到error.jsp这个页面
代码演示:
页面效果:
2、通过web.xml文件指定错误码:响应状态码为404则跳转到404.jsp页面
如果不指定,那么按照以前报错方式,会出现如下错误页面:
我们可以在web.xml进行配置跳转到更加好看一些的页面。
404.jsp页面代码:
3、通过web.xml文件指定错误类型:发生空指针,则跳转到null.jsp页面
demo02.jsp
报空指针异常后的页面:null.jsp
浏览器效果:
JSP动作标签利用XML语法格式的标记来控制Servlet引擎的行为 。利用JSP动作标签可以动态包含其他jsp页面、把用户跳转到另外的页面、为Java插件生成HTML代码。
JSP页面动作标签很多,常用的主要有以下3种:
【作用】
【语法】
【使用示例】
a.jsp
b.jsp
【动态包含和静态包含】
说明:
如果只是引入jsp,不需要改变,那么使用静态引入。如果引入的jsp页面是变化的,那么使用动态引入。
用于页面的转发,与request.getRequestDispatcher(\”/URL\”).foward(request,response);功能一样的
用于给<jsp:forward>和<jsp:include>提供参数
- forward作用
- param的功能
- 功能:用于转发,相当于request.getRequestDispatcher(\”/URL\”).forward(request, response)
- 语法
- 功能:给forward和include提供参数名和值
- 语法:做为子标签存在
- 需求:在c.jsp页面中使用转发标签forward转发到d.jsp页面。
- c.jsp页面
d.jsp页面:
- 从c.jsp转发到d.jsp
- c在请求域中添加键和值,d看能够得到值并且输出
- c转发的时候带参数,username和age,在d中获取并且输出
- 汉字乱码问题的解决
- 转发,url地址栏并没有发生改变
- forward作用:在JSP页面上实现转发
- param的功能:在转发的时候提供其它的参数名和参数值
我们发现,我们在JSP页面中书写的代码最终都会翻译到_jspService()方法中,我们发现这个方法中有两个形参:HttpServletRequest对象 ,HttpServletResponse对象 。所以说,JSP的本质就是一个Servlet。我们可以直接在JSP页面上使用这两个对象。
改造登录后的页面,用户登陆失败以后,跳转到login.jsp,在jsp页面中动态显示用户登陆错误中的信息。
【思路】
- 登录失败后直接转发到一个jsp页面;
- 在jsp页面上使用request对象获取request中的值;
【LoginServlet代码】
说明:LoginServlet中,登录失败之后,直接转发到login.jsp页面。
【login.jsp页面】
说明:上述代码中:
能够使用msg获取数据,因为编译之后存在于同一个类中,编译jsp如下所示:
效果图:浏览器访问login.html页面
登录密码输入错误:
小结:
- JSP作用:给浏览器生成响应信息;
- JSP特点:动态网页,html+java,由服务器来运行的。本质上是一个Servlet;
- 书写Java代码的三种方式:
- 脚本片段:java代码片段,在service方法中起作用。格式:<% System.out.println(\”hello\”) %>;
- 脚本表达式:直接在页面输出内容,在service方法中起作用,等价于out.print(内容)。格式:<%= \”上海打工摸鱼\”%>
- 脚本声明:定义成员变量,方法。在当前类中都起作用。<%!int a=10;%>
- JSP页面常用的对象:request,response;
本文作者及来源:Renderbus瑞云渲染农场https://www.renderbus.com
文章为作者独立观点不代本网立场,未经允许不得转载。