2010-01-26 09:59首先说说为什么要静态化。
对于现在的Web Application来说,动态页面是占有绝对高的地位的,正因为有了动态化,才有了现在Web的丰富多彩,但是如同所有别的事实一样,好处往往都是有代价的。
为了产生出动态的效果,每一次对页面的请求都会要求服务器对页面进行编译或者执行,这些操作都很消耗系统资源。
如果这期间还有和数据库的通讯,那么代价将会更大。
如果一个页面在一定的时间内,其内容没有发生改变,那么就不必为每一次对它的访问进行一次“新”的编译或执行。
我们可以把它在这段没有发生改变的时间内的结果保存到一个静态的页面里面,然后每次访问这个页面时,就用刚才保存的静态页面进行回复。
这样便能大大地减少系统资源的消耗,并且提高对客户的响应速度。
而这个过程就称之为页面静态化。
现在所面临的问题就是如何在JSP中将页面的内容保存下来。
在访问JSP页面的时候,服务器会先将JSP文件编译为Servlet文件,然后对这个Servlet编译、执行,并最终把结果返回给客户端。
而我们的任务就是要把这个最终生成的HTML静态页面保留下来,存放在服务器上。
解决的方法其实很简单。
普通的对于JSP的访问,最终的数据流是回写到客户端的,如果我们把它重定向,让它回写到服务器上一个本地的文件,这样就能把JSP的执行结果保存起来,而这个文件就是我们想要得到的静态化的结果。
好了,说了那么多废话,现在说说具体怎么实现。
看过将JSP编译成的Servlet文件以后,可以知道这个Servlet是通过一个"JspWriter"类型的对象将数据回写给客户端的,而这个对象正是JSP里面的内置对象"out",现在我们只需要定义一个自己的类,让它继承"JspWriter"类,覆盖里面的输出相关函数,让数据都输出到一个指定的文件就可以了。
例如(省略了部分重载函数):程序代码import java.io.*;import javax.servlet.jsp.*;public class ConstPageWriter extends JspWriter {private PrintWriter pw;public ConstPageWriter(String fileName){super(JspWriter.DEFAULT_BUFFER, false);try{pw = new PrintWriter(fileName, "UTF-8");}catch(Exception e){e.printStackTrace();}}@Overridepublic void print(String arg0) throws IOException {pw.print(arg0);}......@Overridepublic void println() throws IOException {pw.println();}@Overridepublic void println(Object arg0) throws IOException {pw.println(arg0);}......@Overridepublic void write(char[] arg0, int arg1, int arg2) throws IOException {pw.write(arg0, arg1, arg2);}}然后将需要静态化的页面写成如下形式:程序代码<%String fileName = "ConstPageName"; // 生成的静态页面文件名if (notNeedUpdate){ // 判断是否需要更新response.sendRedirect(fileName);return;}out = newConstPageWriter(pageContext.getServletContext().getRealPath(fileName) );%><%// Build this page,更新页面%><%out.close();response.sendRedirect(fileName);%>这样页面静态化基本上就算完成了,余下的操作就是处理何时需要更新。
至于何时更新是跟特定的应用程序相关的,这里就不详细分析了。
jsp页面静态化例子2010-05-07 00:35package com.htmlstatic;import java.io.ByteArrayOutputStream;import java.io.File;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.io.OutputStreamWriter;import java.io.PrintWriter;import java.util.ArrayList;import java.util.List;import javax.servlet.RequestDispatcher;import javax.servlet.ServletContext;import javax.servlet.ServletException;import javax.servlet.ServletOutputStream;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import javax.servlet.http.HttpServletResponseWrapper;public class ToHtml extends HttpServlet {private static final String CONTENT_TYPE = "text/html; charset=GBK"; //$NON-NLS-1$private static ServletContext sc = null;private HttpServletRequest request = null;private HttpServletResponse response = null;private static String staticHtmlRefreshTime = "1";//刷新间隔时间private static String path = "D:\\tomcat5.5\\webapps\\diesel_gdc";//文件生成路径private static String url = "/index.jsp";//需要静态化的页面// Initialize global variablespublic void init() throws ServletException {}// Process the HTTP Get requestpublic void doGet(HttpServletRequest request, HttpServletResponseresponse) throws ServletException, IOException {response.setContentType(CONTENT_TYPE);service(request, response);}// Process the HTTP Post requestpublic void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doGet(request, response);}public void destroy() {}public void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {response.setContentType(CONTENT_TYPE);long SPACING = 0;if (staticHtmlRefreshTime == null) {SPACING = 1000 * 60 * 10;} else {SPACING = 1000 * 60 * Long.parseLong(staticHtmlRefreshTime);}sc = getServletContext();this.request = request;this.response = response;String[] urls = url.split(",");List names = new ArrayList();for (int i = 0; i < urls.length; i++) {String tmp = urls[i].replaceAll(".jsp", ".html");names.add(tmp);}while (true) {System.err.println("静态化开始");myService(urls, names);System.err.println("静态化结束");try {Thread.currentThread().sleep(SPACING);} catch (Exception e) {e.printStackTrace();}}}public synchronized void myService(String[] urls, List names) {String name = "";for (int i = 0; i < urls.length; i++) {FileOutputStream fos = null;try {name = path + names.get(i).toString().replaceAll(".html", ".html"); RequestDispatcher rd = sc.getRequestDispatcher("/" + urls[i]);final ByteArrayOutputStream os = new ByteArrayOutputStream();final ServletOutputStream stream = new ServletOutputStream() {public void write(byte[] data, int offset, int length) {os.write(data, offset, length);}public void write(int b) throws IOException {os.write(b);}};final PrintWriter pw = new PrintWriter(new OutputStreamWriter(os)); HttpServletResponse rep = new HttpServletResponseWrapper(response) {public ServletOutputStream getOutputStream() {return stream;}public PrintWriter getWriter() {return pw;}};rd.include(request, rep);pw.flush();System.out.println("name=="+name);fos = new FileOutputStream(name);// 把jsp输出的内容写到xxx.htmos.writeTo(fos);} catch (FileNotFoundException e) {e.printStackTrace();} catch (ServletException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {try {fos.close();} catch (Exception e) {}}}}private void writeFile(String fileName, String content) {OutputStreamWriter writer = null;try {File f = new File(path + "\\" + fileName);if (!f.exists()) {f.createNewFile();}writer = new OutputStreamWriter(new FileOutputStream(f), "GBK"); writer.write(content);writer.flush();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally {try {writer.close();} catch (IOException e) {e.printStackTrace();}}}}?web.xml配置:?<servlet><servlet-name>tohtml</servlet-name><servlet-class>com.htmlstatic.ToHtml</servlet-class></servlet><servlet-mapping><servlet-name>tohtml</servlet-name><url-pattern>*.tohtml</url-pattern></servlet-mapping>?JSP页面静态化总结之二重写JspWriter与采用模版2010-07-13 22:572.1 Tomcat解析后的JSP结构分析从一开始学习Jsp的时候,老师就给我们讲其实JSP最后也是被转变为Servlet 运行的,其源代码可以在Tomcat的work目录中找到。