一、为什么要生成静态页面?

    1.随着你网站的访问量增大,每次访问动态页面都要从数据库中读取一次数据,这无疑是给数据库和服务器端造成了压力与负担。如果没优化的话,往往会导致页面访问速度缓慢。

    2.如果项目程序出现了bug,导致动态页面发生错误不能访问。这时候如果你有生成的静态页面,还是可以访问的,避免带来不必要的损失。

    3.搜索引擎更青睐静态页面,对SEO优化提升搜索排名来说是有益的。(也可使用伪静态优化)


    二、什么地方该生成静态页面呢?

    访问量比较大的页面且内容更新不是较为频繁的。为了用户访问的流畅和服务端的减压,各类门户网站一般是将首页、新闻页等生成静态页面,然后再每隔一段时间自动进行数据更新生成新的静态页面。


    三、如何去实现呢?

    实现的方法有很多,有些人用框架自动生成,有些人重写输出方法实现适合自己的新方法,还有些人用缓存机制从而无需生成静态页面。而我要介绍的方法也是借鉴别人写的,原理就是:后台发送一条指定的链接请求访问servlet的方法,该方法中再发送请求去读取jsp页面动态数据,然后将数据写入到指定的html文件中。说的倒是很轻松,别急,接下来挺我仔细分析:


    1.首先要有个读取jsp数据写入html的主方法(我们发送get请求中需带有的参数 file_name、path、realName、realPath):

/** 
 * @file_name 文件名及文件之后的参数.最好为a.jsf?fileId=aaaa 
 * @path 文件所在的路径.相对于根目录而言的. 
 * @realName 文件要保存的名字 
 * @realPath 文件要保存的真实路径, 默认与文件所在的目录相同。 
 */ 
public class ToHtmlPath extends HttpServlet { 

    public void service(HttpServletRequest request, HttpServletResponse response) 
            throws ServletException, IOException { 

        HttpSession session = request.getSession();     
        String url = ""; 
        String name = ""; 
        System.out.println("name==="+name);
        ServletContext sc = getServletContext(); 

        String file_name = request.getParameter("file_name");// 你要访问的jsp文件,如news.jsf。 
        // file_name如:fileDetail.jsf?fileId=56.要是有参数, 只有一个参数。并且以参数名作为文件名。 

        String realName = request.getParameter("realName");// 要保存的文件名。如aaa;注意可以没有这个参数。 

        String path = request.getParameter("path");// 你要访问的jsp文件路径。如news。注意可以没有这个参数。 

        String realPath = request.getParameter("realPath");// 你要保存的文件路径,如htmlNews.注意可以没有这个参数。 
        // 下面确定要保存的文件名字。 
        if (realName == null || realName == "") { 
            int a = 0; 
            a = file_name.indexOf("=") + 1; 
            realName = file_name.substring(a); 
            if (realName.indexOf(".")>0) { 
                realName = file_name.substring(0, file_name.indexOf(".")); 
            } 
        } 
        // 下面构造要访问的页面。 
        if (path == null || path.equals("")) { 
            url = "/" + file_name;// 这是你要生成HTML的jsp文件,如 
        } else { 
            url = "/" + path + "/" + file_name;// 这是你要生成HTML的jsp文件,如 
        } 
        // System.out.println("url==="+url);
        // 下面构造要保存的文件名,及路径。 
        // 1、如果有realPath,则保存在realPath下。 
        // 2、如果有path则保存在path下。 
        // 3、否则,保存在根目录下。 
        if (realPath == null || realPath.equals("") ) { 
            if (path == null || path.equals("") ) {
                // 这是生成的html文件名,如index.htm.说明: ConfConstants.CONTEXT_PATH为你的上下文路径。 
                name = session.getServletContext().getRealPath("") + "\\" + realName + ".html";
            } else { 
                name = session.getServletContext().getRealPath("") + "\\" + path + "\\" 
                        + realName + ".html";// 这是生成的html文件名,如index.html 
            } 
        } else { 
            name = session.getServletContext().getRealPath("") + "\\" + realPath + "\\" 
                    + realName + ".html";// 这是生成的html文件名,如index.html
        } 

        // 访问请求的页面,并生成指定的文件。 
        RequestDispatcher rd = sc.getRequestDispatcher(url); 

        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; 
            } 
        }; 
        rep.setCharacterEncoding("gbk");//response的编码为gbk防乱码
        rd.include(request, rep); 
        pw.flush(); 
        FileOutputStream fos = new FileOutputStream(name); // 把jsp输出的内容写到xxx.html 

        os.writeTo(fos); 
        fos.close(); 
        
    } 
}


    2.需要一个方法来调用上面的主方法,同时这个方法即是后台发送请求访问servlet的方法:

public class CallJspToHtml {

    //这个方法适当重载,就可以省去一些参数传递。
    public static void CallOnePage(String basepath,String fileName, String path, 
            String realName, String realPath) { 
           try { 
            String str = basepath+"toHtmlPath?file_name=" 
              + fileName + "&path=" + path + "&realName=" + realName 
              + "&realPath=" + realPath; 
            int httpResult; //请求返回码

            URL url = new URL(str); //URL发送指定连接请求
            URLConnection connection = url.openConnection(); 
            connection.connect(); 
            HttpURLConnection httpURLConnection = (HttpURLConnection) connection; 
            httpResult = httpURLConnection.getResponseCode(); 
            
            if (httpResult != HttpURLConnection.HTTP_OK) { //返回码为200则连接成功
             System.out.println("没有连接成功!"); 
            } else { 
             System.out.println("连接成功了!!!!!"); 
            } 
           } catch (Exception e) { 
               e.printStackTrace();
           } 
          } 
}

    上面2的方法发送的请求又是如何调用到3的servlet中方法呢?


    3.web.xml配置

<servlet> 
      <servlet-name>toHtmlPath</servlet-name> 
      <servlet-class>com.tptravel.util.compilerhtm.ToHtmlPath</servlet-class> 
  </servlet> 
  <servlet-mapping> 
      <servlet-name>toHtmlPath</servlet-name> 
      <url-pattern>/toHtmlPath</url-pattern> 
  </servlet-mapping>

    嗯,没错。就是在配置文件中指定请求链接跳转至相应的方法,这样2中所发送的请求就会相应的去找到3中的方法。每个人使用框架都不一样,所以发送的链接请求样式也不一样,只要你请求与参数相对应就行,其实都是一个道理。

    

    举个请求链接的例子分析:http://localhost:8080/JspToHtml/toHtmlPath?file_name=jnh_showpage/show_info.do?tp=newsDetail&ids=1404955281632&path=&realName=1404955281632&realPath=html\news

    红色部分就是你所要转换的jsp动态页面的访问链接

    ***部分为要转换的jsp文件所存放位置,因为红色部分我们就已经可以访问到了jsp所以可以省略

    蓝色部分为生成的静态文件的名字

    绿色部分为静态文件所要存放的位置(默认是在项目更目录下),例子中则是存放在根目录的html文件夹中的news文件夹中。


    4.具体应用分析:

    把上述的方法都封装到自己的常用Util类中。

    手动生成:后台在发布新闻、更新新闻,生成动态文件的同时则可以调用CallJspToHtml.CallOnePage(对应参数)生成html静态文件。

    自动生成:设置监听器去判断其若超出约定时间则自动更新html文件。


    结语:分享是一种快乐,坚持是一种精神。