我们都知道在jsp中include有两种形式,分别是Include指令:<%@ include file=""%>和include动作:<jsp:include page="" flush="true"/>
前者是指令元素、后者是行为元素。具体它们将在何处用?如何用及它们有什么区别?这应该是很多人看到它都会想到的问题。下面一起来看看吧。
通常当应用程序中所有的页面的某些部分(例如标题、页脚和导航栏)都相同的时候,我们就可以考虑用include。具体在哪些时候用<%@ include file=""%>,哪些时候用<jsp:include page="" flush="true"/>。这种形式。首先要明白的是它们之间的区别。只有了解了它们用法的不同才理解该在何时去用以及如何选择。
两者最重要的区别:JSP指令<%@ include file=""%>,是将被引入的JSP与原JSP融合到一起,而这个融合过程是在翻译阶段进行的。
为什么需要翻译阶段?我们知道,jsp页面并不是原封不动的发送到客户端的,因为浏览JSP页面的客户端并不需要安装Java虚拟机,客户端机器并不能读懂JSP,它能读懂的只有HTML、JavaScript (当然还有其他,例如:Applet、Flex、AxtiveX等等,但那些都需要下载相应的客户端解析器),这样就需要Servlet Engine (例如:Tomcat) 将所有的JSP元素进行处理。这是通过将jsp页面转化成Servlet,然后执行这个Servlet来完成的。服务器需要一个jsp容器来处理jsp页面。jsp容器通常以Servlet的形式实现,这个servlet经过配置,可以处理对jsp页面的所有请求。
JSP容器负责将jsp页面转化成servlet,并编译这个servlet。这两个步骤就构成了翻译阶段。
而jsp翻译之后的servlet输出的内容才是客户端浏览器能够识别的东西,HTML、JavaScript之类的,servlet是使用JspWriter对象输出输出这些HTML、JavaScript的。如果你去翻看翻看jsp编译后的servlet代码,你会发现很有意思的东西,比如Struts的<logic:iterator>标签,被翻译成do{}while()语句实现循环。如果我们把<bean:write>写在<logic:iterator>内部,则在do的内部会出现类似_jspx_meth_bean_write_2(_jspx_th_logic_iterate_0, _jspx_page_context)的方法调用。这些说明了一切。
由此我们知道:jsp页面是把include指令元素(<%@ include file=""%>)所指定的页面的实际内容(也就是代码段)加入到引入它的jsp页面中,合成一个文件后被jsp容器将它转化成servlet。可以看到这时会产生一个临时class文件和一个servlet源文件。而动作元素(<jsp:include page=""/>)是在请求处理阶段引入的,会被JSP容器生成两个临时class文件和两个servlet原文件。而引入的只是servlet的输出结果,即JspWriter对象的输出结果,而不是jsp的源代码。
举个例子:
main.jsp
- <%@ page language="java" pageEncoding="GBK"%>
-
- <%@ taglib uri="http://struts.apache.org/tags-bean" prefix="bean" %>
- <%@ taglib uri="http://struts.apache.org/tags-html" prefix="html" %>
- <%@ taglib uri="http://struts.apache.org/tags-logic" prefix="logic" %>
- <%@ taglib uri="http://struts.apache.org/tags-tiles" prefix="tiles" %>
-
-
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
- <html:html lang="true">
- <head>
- <html:base />
-
- <title>index.jsp</title>
- <meta http-equiv="Content-Type" content="text/html; charset=GBK" />
- <meta http-equiv="pragma" content="no-cache">
- <meta http-equiv="cache-control" content="no-cache">
- <meta http-equiv="expires" content="0">
- <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
- <meta http-equiv="description" content="This is my page">
-
-
-
-
- </head>
-
- <body>
-
- <%-- <%@ include file="include/head.jsp"%>--%>
- <jsp:include page="include/head.jsp"></jsp:include>
-
- <%-- <%@ include file="include/menubar.jsp"%>--%>
- <jsp:include page="include/menubar.jsp"></jsp:include>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- 我是主体<br><br><br>
- <%-- <%@ include file="include/copyright.jsp"%>--%>
- <%-- <jsp:include page="/includeSample_copyright.do"></jsp:include>--%>
- <%-- <jsp:include flush="true" page="include/copyright.jsp"></jsp:include>--%>
- </body>
- </html:html>
head.jsp
- <%--<%@ page language="java" pageEncoding="GBK"%>--%>
-
- <%--<%@ taglib uri="/WEB-INF/struts-html.tld" prefix="html" %>--%>
- <%--<%@ taglib uri="/WEB-INF/struts-logic.tld" prefix="logic" %>--%>
- <%--<%@ taglib uri="/WEB-INF/struts-bean.tld" prefix="bean" %>--%>
-
- <table>
- <tr>
- <td><bean:message key="copyright.inc.copyright"/>
- 我是Head 我是Head我是Head我是Head我是Head我是Head我是Head</td>
- </tr>
- </table>
以head.jsp为例:
1、如果我是用<%@ include file="include/head.jsp"%>引入,注意,head.jsp被我注释掉的字符集和Struts标签的引入,如果打开注释,会怎么样呢??会抛出500异常,/main.jsp(44,4) /include/head.jsp(3,56) Attempt to redefine the prefix html to /WEB-INF/struts-html.tld, when it was already defined as http://struts.apache.org/tags-html in the current scope.这时因为在翻译阶段main.jsp和head.jsp被原封不动的合称为一个jsp,察看Tomcat工作目录只有一个servlet类文件。
\work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.java
\work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.class
试想,在一个类文件中两次引入相同的Struts标签,编译时当然回抛出异常了。
2、那么如果我是以<jsp:include page="include/head.jsp"/>方式引入呢?首先先将head.jap中注释的Struts标签的部分打开,而字符集部分仍然注释。结果出现乱码:
- 结果:
- 2004-2006 版权所有
- ????Head ????Head????Head????Head????Head????Head????Head
- ????MenuBar
- 我是主体...
什么原因?include动作元素是在请求阶段执行引入的,所以它引入的只是head.jsp被翻译成servlet文件中_jspService这个方法中JspWriter这个对象的输出(out.write()方法的输出流)。该输出的执行是在head.jsp被引入main.jsp之前就进行了,所以main.jsp页面中的字符集设置当然对head.jsp不起作用了。
如果Struts标签部分也注释掉呢?"2004-2006 版权所有"这一行不会输出,因为这一行是由Struts标签输出的,没有输出的原因和字符集相同,我想大家应该明白了。
最后观察Tomcat工作目录下,会有两个Servlet:
第一个:\work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.java
\work\Catalina\localhost\IncludeAction\org\apache\jsp\main_jsp.class
第二个:\work\Catalina\localhost\IncludeAction\org\apache\jsp\include\head_jsp.java
\work\Catalina\localhost\IncludeAction\org\apache\jsp\include\head_jsp.class
另外,如果希望通过修改后缀的方法表示哪些是被引入的文件,例如:将head.jsp改名为head.inc的话,JSP容器不能识别*.inc。所以不能翻译head.inc,所以此时只能使用<%@ include file="head.inc"%>方法引入一个文件了。
我们来总结一下两种include 两种用法的区别,主要有两个方面的不同:
一、执行时间上:
<%@ include file=”relativeURI”%> 是在翻译阶段执行
<jsp:include page=”relativeURI” flush=”true” /> 在请求处理阶段执行。
二、二:引入内容的不同:
<%@ include file=”relativeURI”%> 引入静态文本(html,jsp),在JSP页面被转化成servlet之前和它融和到一起。
<jsp:include page=”relativeURI” flush=”true” /> 引入执行页面或servlet所生成的应答文本。
分享到:
相关推荐
在JavaServer Pages (JSP) 技术中,`<jsp:include>` 和 `<%@ include %>` 是两个用于页面组合的指令,它们虽然都用于将一个或多个文件的内容插入到主页面中,但它们的工作机制和使用场景有所不同。理解这两者的区别...
<%-- 或者使用静态包含:<%@ include file="date.jsp" %> --%> </head> <body> <p>今天的日期是:</p> </body> </html> ``` 如果使用`<jsp:include page="date.jsp" flush="true"/>`,那么每当访问`test.jsp...
JSP 中的 Include 有两种用法,分别是 `<%@ include file=” ”%>` 和 `<jsp:include page=” ” flush=”true”/>`。这两种用法都可以用于引入其他 JSP 文件,但是它们之间存在着一些关键的区别。 首先,让我们...
<td height="277" align="center" valign="top"><%@include file="changxiao.jsp"%></td> </tr> </table> <br> <table width="208" height="356" border="0" cellpadding="0" cellspacing="0" background=...
nt test</title></head><body>This content is statically in the main JSP file.<br/><jsp:include page="included.html"/></body></html>]]>)正如你所见,清单2使用了传统的`<%@include file="..."%>`伪指令来包含...
<%@ page language="java" contentType="text/html; charset=gb2312"%> <html> ... <tr ><td colspan="2"><%@ include file="view/AdminEnd.jsp" %></td></tr> </table> </center> </body> </html>
jsp:include 动作与`<%@ include %>`类似,但可以动态地包含页面。 **JSP语法**: ```jsp <jsp:include page="page.jsp" flush="true"/> ``` **示例**: ```jsp <jsp:include page="header.jsp" flush="true"/> ``` ...
- `<jsp:include page="header.jsp" />` 和 `<jsp:include page="footer.jsp" />` 会在运行时动态加载`header.jsp`和`footer.jsp`的内容。 - 这种方式称为动态包含,因为包含的动作是在运行时发生的,而不是在编译时...
<%@ include file="../common/header.jsp" %> ``` - **页面实际内容**: ```jsp <%--页面实际内容 --%> <p>本实例向您展示了使用 include 指令的方法!</p> ``` - **包含页脚文件内容**: ```jsp <%--包含...
1. **编译与运行时的区别**:`<%@ include file="% %>`在编译时执行,而`<jsp:include>`在运行时执行,这决定了它们在性能和灵活性上的不同表现。 2. **数据流管理**:使用`<jsp:include>`时,需要注意被包含页面...
根据提供的文件信息,本篇文章将围绕“在JSP页面中包含文件”的主题展开,深入探讨JSP中的两种主要包含机制:`jsp:include`动作标签与`<%@ include %>`指令,以及它们在实际开发中的应用场景。 ### JSP页面中的文件...
<%@ include file="header.jsp" %> ``` **2.2 动作元素** 动作元素允许开发者在运行时动态地插入文件、重定向请求、收集用户输入等,主要的动作元素包括: - **jsp:include**:动态地包含另一个文件。 - **jsp:...
<%@ include file="conn.jsp"%> <% sql="select * from users where name='"+name+"' and password='"+password+"'"; ResultSet rs =statement.executeQuery(sql); if (rs.next()) { disable=rs.getBoolean(...
然后在`content.jsp`中使用`<%@ include file="...">`指令包含头部和尾部的文件。 **案例实现**: 1. **编写head.jsp**:包含网页的头部信息,如Logo、导航条等。 ```jsp <%@ page language="java" contentType...
* 例子:`<%@ page language="java" %> <html> <head><title>A Comment Test</title></head><body> <h2>A Test of Comments</h2> <%-- This comment will not be visible in the page source --%> </body> </html>` ...
这与JSP的`<jsp:include>`动作元素不同,后者是在运行时动态地包含页面内容。 理解并正确使用JSP指令是成为一名熟练的JSP开发者的关键。通过这些指令,开发者可以更好地控制页面行为,优化性能,以及实现更复杂的...
<%@ include file="check.jsp"%> <html> <frameset rows="15%,*"> <frame name="top" src="top.jsp" frameborder="1" bordercolor="#FFCCCC"> <frameset cols="30%,*"> <frame name="left" src="left....
当使用 `<%@ include file="date.jsp" %>` 或 `<jsp:include page="date.jsp" flush="true"/>` 方式包含时,主页面和被包含的页面都设置了 `contentType` 导致冲突。 为了解决这个问题,我们可以采取以下几种方法:...