JSP2.0技术手册中一个例子给我的困惑

作者讲解 javabean的范围的时候,特别举了个计数器的例子,一个javabean叫Counter.java, 然后在Counter.jsp中调用这个bean,同时Counter.jsp中include了一个Hello.jsp。应为Counter.jsp调用 bean的时候,设置的scope是 page,所以Hello.jsp没法使用 Counter.jsp调用的bean,这个没有问题。但是作者说,如果把 scope,改为 request(session,application)后,Hello.jsp就可以调用了。我做了实验,发现改变scope再republish后,Hello.jsp还是无法调用呢?抛出异常为“Attempted a bean operation on a null object.”

另外,如果我在Hello.jsp中再次调用 ,Hello.jsp就能显示了。

代码如下:

//Counter.java
package com.javastudy.ch8;

public class Counter {

public Counter(){

}
private int count = 0;
public int getCount(){
count++;
return count;

}
public void setCount(int newCount){
count = newCount;
}

}

//Counter.jsp
<%@ page contentType="text/html;charset=Big5" %>



CH8 - Counter1.jsp

範圍為 Request的 JavaBean範例程式 - Counter1

//用session,request,application,hello.jsp都无法使用

您己參觀 次

歡迎再次參觀

//Hello.jsp

<%@ page contentType="text/html;charset=Big5" %>
Hello ... 您好

感謝您第次的光臨
[b]问题补充:[/b]
不好意思,我复制到这上面来的时候,搞错了,应该是

这个不是我问题的错误所在。
[b]问题补充:[/b]
回楼下的Rowin,这个异常应该是跟Bean的作用域有关,我把scope设置成page,抛出这个异常可以理解,但是我把作用域设置成大到application都仍然有这个问题,我就没法理解了。是不是环境的问题哦?
[b]问题补充:[/b]
回lovewhzlq,我用的tomcat是apache-tomcat-5.5.27.我遇到的问题和下面这个帖子中的问题一模一样

http://www.javaworld.com.tw/jute/post/view?bid=22&id=59329
[b]问题补充:[/b]
回panjava,我把 换为

/jsp:include

就抛新的异常了。。。。
ervlet.service() for servlet jsp threw exception
org.apache.jasper.JasperException: /Counter1.jsp(19,0) Expecting "jsp:param" standard action with "name" and "value" attributes

到底怎么回事啊? 我是在eclipse里建了个 dynamic web Project,然后绑定了tomcat为server.Tomcat 5.5.27, JDK1.6,Eclipse是最新的。
[b]问题补充:[/b]
书中代码,我上传上来了。那个Counter.java要自己写下。
[b]问题补充:[/b]
对了,我有个发现,把 换成 <%@ include file="Hello.jsp"%>,就一切正常了。不晓得是什么原因。
[b]问题补充:[/b]
下面的链接中的人,也遇到了我同样的问题
http://topic.csdn.net/u/20090219/13/5c33d2a1-dbcc-4347-b643-72ee73d8f2d2.html
[b]问题补充:[/b]
谢谢楼下的大哥们,小弟刚学J2EE,这两种的区别还不太了解,等下仔细看看你们的资料,难道是JSP2.0书上写错了?先做事去了,等下回来给分,谢谢各位。

8个回答

这两种方式是不同的

< %@ include file=” ”%>

< jsp:include page=” ” flush=”true”/>

  前者是指令元素、后者是行为元素。具体它们将在何处用?如何用及它们有什么区别?这应该是很多人看到它都会想到的问题。下面一起来看看吧。

  通常当应用程序中所有的页面的某些部分(例如标题、页脚和导航栏)都相同的时候,我们就可以考虑用include。具体在哪些时候用< %@ include file=” ”%>,哪些时候用< jsp:include page=” ” flush=”true”/>.这种形式。首先要明白的是它们之间的区别。只有了解了它们用法的不同才理解该在何时去用以及如何选择。

  < %@ include file=” ”%>,jsp的include指令元素读入指定页面的内容。并把这些内容和原来的页面融合到一起。(这个过程是在翻译阶段:也就是jsp被转化成servlet的阶段进行的。

  这里对翻译阶段进行一下说明:我们知道,jsp页面不能原封不动地被传送给浏览器,所有的jsp元素都必须首先由服务器进行处理。这是通过将jsp页面转达化成servlet,然后执行这个servlet来完成的。服务器需要一个jsp容器来处理jsp页面。jsp容器通常以servlet的形式来实现,这个servlet经过配置,可以处理对jsp页面的所有请求。

  Jsp容器负责将jsp页面转化成servlet(称为jsp页面实现类?JSP Page implementation class),并编译这个servlet。这两步就构成了翻译阶段.

  由此我们就会知道:jsp页面是把include指令元素所指定的页面的实际内容(也就是代码段)加入到引入它的jsp页面中,合成一个文件后被jsp容器将它转化成servlet。可以看到这时会产生一个临时class文件和一个java文件。下面举个例子。

  服务器用tomcat,引入页面的jsp文件叫test.jsp。被引入的页面叫date.jsp.这个jsp文件里存放的是一个有关时间的jsp代码,当前的上下文根设为test

//======date.jsp的源文件=====//
<%@ page language=”java” contentType="text/html;charset=gb2312"%>
<%
java.util.Date date=new java.util.Date();
String date_cn ="";
String dateStr = "";
switch(date.getDay())
{
case 0:date_cn ="日"; break;
case 1:date_cn ="一"; break;
case 2:date_cn ="二"; break;
case 3:date_cn ="三"; break;
case 4:date_cn ="四"; break;
case 5:date_cn ="五"; break;
case 6:date_cn ="六"; break;
}
dateStr = (1900+date.getYear())+"年"+(date.getMonth()+1)+"月"+date.getDate()+
"日(星期" + date_cn + ")";
%>
document.write("<%=dateStr%>");
//======以下是test.jsp的源文件=============//
<%@ page language=”java” contentType=”text/html;charset=gb2312”%>


include的两种用法

<%--@ include file=”date.jsp” %-->
//我们在这里用include的两种不同形式来引入date.jsp这个文件.

有关jsp中include的两种用法.敬请关注。


  在test.jsp 文件中,我们只输出了一行文本“ 有关jsp中include的两种用法.敬请关注。”,现在让我们先用这种形式引入date.jsp这个文件。你想会出现什么问题了吗?此时出现了错误提示:

HTTP Status 500 ?
org.apache.jasper.JasperException: /date.jsp(0,0) Page directive:
can't have multiple occurrences of contentType

  以下还有一堆错误,但我们只要看这里就知道问题的所在了。状态码为http 500服务器内部错误。再看下面的提示。在date.jsp页面中不能指定多个contentType.

原因就在这里了。是因为在翻译阶段,date.jsp文件的代码被原封不动地加入到了test.jsp页面从而合成一个文件。合成后的文件中就会相同的:

< %@ page language=”java” contentType=”text/html;charset=gb2312”%>

  这句代码。解决的办法是把date.jsp文件中的这句删掉。刷新后再请求test.jsp页面

请求test.jsp在页面显示如下

2007年8月12日 13:12:40

有关jsp中include的两种用法.敬请关注。

  这时我们还不能发现什么。还是去查看tomcat下的临时文件吧。到那里去看看date.jsp文件的内容是否已被加入到了test.jsp文件中。

< 注.此处的tomcat装在E盘根目录下>

目录

E:\tomcat\work\Standalone\localhost\test.

在这个目录下会看到

test_jsp.java和test_jsp.class两个文件。

  这里的java文件就是jsp容器将jsp转化成了servlet而得到的test_jsp.java这个文件。

  相对应的test_jsp.class这个文件就是编译test_jsp.java这个servlet文件产生的类文件了。打开所产生的servlet 文件(test_jsp.java)。此时我们会发现,在test.jsp 文件被转化成servlet文件时,在输出的< haed >之间加入了一些不是test.jsp页面里面的代码,新加入的内容就是 date.jsp里面的代码: 新加入了哪些内容或是否真的加入了新的内容请自己测试去看一下就会一目了然了.在这里不再详述.

以上就是我们用< %@ include file=”date.jsp”%>这种形式得到的结果.

下面我们换用< jsp:include page=”dae.jsp” flush=”true”/>也就是将

< %@ include file=”date.jsp”%>换成< jsp:include page=”dae.jsp” flush=”true”/>,然后请求test.jsp.

2007? ê 8??12?? 13:30:13

  有关jsp中include的两种用法.敬请关注。

  此时会在页面上看见.我们所引入date.jsp输出的日期中中文出现了乱码.什么原因?是因为include行为元素是在请求处理阶段执行的(此处要对请求处理阶段进行说明一下.Jsp容器除了上面提到的负责将jsp页面转化成servlet外,还负责调用jsp页面实现类以处理每个请求并产生应答. 这个阶段我们就称为请求处理阶段.请求处理阶段仅执行类文件)。

  所以在我们作include行为元素引入页面时,实际只是引用了 date.jsp这个文件被转化并被编译后产生的servlet类文件.既如此, date.jsp就是作为一个单独的文件在执行后才被test.jsp文件运行时调用.由于date.jsp文件中没有指定字符编码.所以出现了乱码.解决办法是在date.jsp文件中重新把刚才去掉的

  这行语句加入后刷新重新运行.此时页面显示正确,并跟用include指令正常运行时一样.再查看tomcat下的临时文件会发现.此时多出了一个 date_jsp.java文件和一个date_jsp.class文件.这两个文件得来的方式跟test_jsp.java和 test_jsp.class文件得来的方式一样.再查看一下此时test_jsp.java文件的代码会发现.此时只新增加了一句代码:

JspRuntimeLibrary.include(request, response, "date.jsp", out, true);

  它并没有把date.jsp文件的代码加入到test.jsp.

  只是在运行时引入了date.jsp页面执行后所产生的应答.这意味着我们可以指定任何能够产生应答的Web资源,(例如一个servlet或一个 jsp页面),只要这些资源所产生的类型和jsp页面产生的内容类型相同.JSP容器将通过一个内部的函数调用来执行指定的资源.因此,这些被引入的资源可以帮助处理原来的请求,所以这些资源可以访问请求作用域内的所有对象.以及所有原来的请求参数.

  由于在主页面被请求时,这些页面还没有被引入到主页面中,所以你可以对page属性使用一个请求时属性值,以便根据运行时的情况来决定要引入哪一个页面.还可以添加一些将被引入的页面读取的请求参数.

” flush=”true” >


/jsp:include

  如果修改了被引入的jsp页面,那么可以立刻使用该页面的最新版本,这是因为对待被引入的页面的方式与对待由浏览器直接调用的jsp页面的方式完全相同.即容器检测页面的变化,并自动进入翻译阶段,以得到页面的最新版本.

(注意,include行为元素同jsp其它元素一样,没有行为体时要以”/”结束.就像下面这样.

< jsp:include page=”” flush=”true” />

以下是对include 两种用法的区别

主要有两个方面的不同;

  1.执行时间上:

< %@ include file=”relativeURI”%> 是在翻译阶段执行

< jsp:include page=”relativeURI” flush=”true” /> 在请求处理阶段执行.

  2.引入内容的不同:

< %@ include file=”relativeURI”%>

  引入静态文本(html,jsp),在JSP页面被转化成servlet之前和它融和到一起.

< jsp:include page=”relativeURI” flush=”true” />引入执行页面或servlet所生成的应答文本.

  另外在两种用法中file和page属性都被解释为一个相对的URI.如果它以斜杠开头,那么它就是一个环境相关的路径.将根据赋给应用程序的URI的前缀进行解释,如果它不是以斜杠开头,那么就是页面相关的路径,就根据引入这个文件的页面所在的路径进行解释.

朋友,你自己写错了
你都没写对类名,只写了包名啊???!!!

应该是

楼主有没查过下面这个异常..
[quote]Attempted a bean operation on a null object[/quote]

Hello.jsp抛出这个异常,是因为对NULL对象进行bean操作...

我觉得没可能,我以前都跑过这里的例子,
你把你的例子上传上来,

对了,你用的是tomcat什么版本,

[quote]是不是环境的问题哦? [/quote]
之前我查这个异常的时候有看到别人提到环境的问题...你去google一下看看吧..

/jsp:include
用这个试一下.

[code="jsp"]<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@page import="com.parse.counter.Counter"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">



感謝您第次的光臨

[/code]

[code="jsp"]<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">


/jsp:include


[/code]

没有问题啊 j2ee4

不一样的东西:
<jsp:include //行为元素

<%@ include //指令元素

[url]http://java.chinaitlab.com/ServletJsp/529740.html[/url]
[url]http://developer.51cto.com/art/200902/111134.htm[/url]

Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
立即提问