一个Jsp初学者的学习过程(五)

来源:互联网 发布:冒险岛台服数据库 编辑:程序博客网 时间:2024/05/22 17:11
一个Jsp初学者的学习过程(五)

TheUnforgiven


第五章  分页功能的实现

    在我逐步把我的留言板的功能完善的同时,我渐渐熟悉了对数据库的操作,这时我发现留言信息的目录越来越长了,我需要实现一个分页功能了,最初我尝试自己解决这个问题:
    1、我应该把它的关键部分封装成一个bean,使它尽可能的能够重用;
    2、通过资料了解有两种数据库查询方案:一、一次取得所有资料,然后在指定的页显示指定的资料;二、分次查询数据库,每次只获得本页的数据。考虑到数据库中记录数越多,方案一所占的服务器资源就越多(将所有的记录都放到内存中,假如有50万条记录的话……@$^*%@#),所以应该采用方案二;
    3、首先要知道目标数据库里共有多少条记录(select count(*) from 表名),然后确定每页显示多少条记录,再根据它计算一共分多少页(最大页数)显示,这部分由bean1(在我的代码里就随便起个名叫PageBean)实现;获取当前要显示第几页的请求,查询本页要显示哪些条记录,将每条记录的内容作为一组数据返回给显示页面,这部分由bean2(CountBean)实现;显示页面(.jsp文件)显示各条记录的内容。
    到这里我发现有两个难点:(1)查询从第m条到第n条记录的SQL语句不会写,通过在网上查找资料,这个问题得到解决;(2)bean2返回给显示页面的值是个二维的数组,这个数组怎么传呢?于是不得不上网翻书查找资料,最终发现由Vector(向量)来解决,由于没有工具书,无法针对Vector进一步学习,就只好分析代码,好在最后分析明白了——到目前为止,也只是明白了那段代码,仍然无法做到能够应用。(后补:这些东西事实上也不是难于理解的,有时候我们需要做的是在有了一定理论基础之后查看文档和别人的说明。在别人的指点下,我现在的代码里边基本都不使用Vector了,采用ArrayList来代替它,实际上,还有人建议我使用Hashtable代替ArrayList)
    下面是这三个文件的代码:
-----------------------------------------PageBean.java-----------------------------------------------
//该bean用于实现分页功能时得到总的记录数和最大的页数
package ringz.javabeans;
import java.io.*;
import java.sql.*;
public class PageBean
{
   private int maxRowCount;//最大记录数
   private int onePageRowCount;//每页显示的记录数
   private int maxPageCount;//最大页数
   private String classforname;
   private String servanddb;
   private String sql;
   //
   //得到关于目标数据库的搜索条件
   public void setSql(String s1,String s2,String sql)
   {
     this.classforname=s1;
     this.servanddb=s2;
     this.sql=sql;
   }
  
   //得到onePageRowCount
   public void setOnePageRowCount(int counts)
    {
      onePageRowCount=counts;
    }

   //计算maxRowCount并返回
   public int getMaxRowCount()throws Exception
    {
     try
     {
     Class.forName(classforname);//载入驱动程式类别
     Connection con=DriverManager.getConnection(servanddb);//建立数据库连接
     Statement stmt=con.createStatement();//建立Statement变量
         ResultSet rs=stmt.executeQuery(sql);
     if (rs.next())
       maxRowCount=rs.getInt(1);
     rs.close();
     stmt.close();
     con.close();
     return maxRowCount;
     }//try
     catch (Exception e)
     {
      e.printStackTrace();
      throw e;
      }
    }//getMaxRowCount()
 

   //根据maxRowCount和onePageRowCount计算出maxPageCount并返回
   public int getMaxPageCount()
    {
      if (maxRowCount%onePageRowCount==0)
        maxPageCount=maxRowCount/onePageRowCount;
      else
        maxPageCount=maxRowCount/onePageRowCount+1;
      return maxPageCount;
    }
 }
---------------------------------------------------------------------------------
----------------------------CountBean.java------------------------------------
//该bean用于接收具体页数然后返回该页应显示的记录
package ringz.javabeans;
import java.util.*;
import java.io.*;
import java.sql.*;
public class CountBean
{
   private int pageNum;//当前是第几页
   private String classforname;
   private String servanddb;
   private String sql;
   Vector v=new Vector();
  
   //得到关于目标数据库的搜索条件
   public void setSql(String s1,String s2,String sqlstr)
   {
        this.classforname=s1;
     this.servanddb=s2;
     this.sql=sqlstr;
    }
  
   //得到pageNum
   public void setPageNum(int pagenum)
    {
      this.pageNum=pagenum;
     }
     
   //返回结果
   public Vector getResult(String listname[])throws Exception
    {
     int num=listname.length;//得到数组的长度
     String listName[]=new String[num];//定义一个大小为num的string型数组
     for(int i=0;i<num;i++)
       listName[i]=listname[i];//将目标数组的内容传给listName数组
     
     try
     {
     Class.forName(classforname);
     Connection con=DriverManager.getConnection(servanddb);
     Statement stmt=con.createStatement();
        ResultSet rs=stmt.executeQuery(sql);
     int i=1;
     while(rs.next())
      {
        Object[] obj=new Object[num];
        for(int j=0;j<num;j++)
          obj[j]=rs.getString(listName[j]);
        v.add(obj);
        i++;
       }//while
     rs.close();
     stmt.close();
     con.close();
     return v;
     }//try
     catch(Exception e)
     {
      e.printStackTrace();
      throw e;
      }//catch
    }
}
----------------------------------------------------------------------------------
    编译上面这两个文件的时候,发现如果不把“错误”扔掉(throw e),就无法编译成功,我不明白具体原因。
-----------------------------------page.jsp---------------------------------------
<%@ include file="include.inc"%>
<%@ page contentType="text/html;charset=gb2312"%>
<jsp:useBean id="page1" scope="page" class="ringz.javabeans.PageBean"/>
<jsp:useBean id="page2" scope="page" class="ringz.javabeans.CountBean"/>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312">
<title>分页的实现</title>
<style type="text/css">
<!--
body {
    margin-left: 10%;
    margin-right: 10%;
     }
-->
</style>
<script language="JavaScript" type="text/JavaScript">
<!--
function jumping(jump){ //v3.0   ***用于页面跳转
  var pageID=jump.value;
  var url=pageID;
  window.location.href = url;
}
//-->
</script>
</head>

<body>
<%
int pageNum;
try
{
  pageNum=Integer.parseInt(request.getParameter("page"));//得到“要显示第几页”
}
catch(Exception e)
{
  pageNum=1;//如果出错说明pageNum没有接收到"page",那么就把pageNum初始为1
}
int onePageRowCount=10;//每页的条数************按需要改变
String s1="oracle.jdbc.driver.OracleDriver";//定义载入驱动程式的字符串
String s2="jdbc:oracle:thin:name/password@ringz:1521:rock";//定义建立数据库连接的字符串************按需要改变
String sql="select count(*) from article";//确定记录总数的查询语句************按需要改变
page1.setSql(s1,s2,sql);
page1.setOnePageRowCount(onePageRowCount);
int maxRowCount=page1.getMaxRowCount();//获得记录总数
int maxPageCount=page1.getMaxPageCount();//获得总的页数
page2.setPageNum(pageNum);
String listname[]={"ID","author","time","title"};//要查询的字段名************按需要改变
int max=pageNum*onePageRowCount;//本页最后一条记录的行号
int min=(pageNum-1)*onePageRowCount+1;//本页第一条记录的行号
String sqlstr="select b.* from (select a.*,rownum row_num from (select * from article order by time desc) a where rownum<='"+max+"') b where row_num>='"+min+"'";//************按需要改变
page2.setSql(s1,s2,sqlstr);
if(maxRowCount>0)
  out.println("<div align='left'>共有"+maxRowCount+"条记录,每页显示"+onePageRowCount+"条。</div>");
%>
<table width="100%" border="1" align="center" cellpadding="0" cellspacing="0" bordercolorlight="#000000">
<tr bgcolor="#00CCFF">
<td align="center">标题</td>
<td align="center">作者</td>
<td align="center">日期</td>
</tr>
<%
java.util.Vector v=page2.getResult(listname);
java.util.Enumeration e=v.elements();
while(e.hasMoreElements())
{
 Object[] obj=(Object[])e.nextElement();//****************注意修改下面的几行
 String id=obj[0].toString();
 String name=obj[1].toString();
 String time=obj[2].toString();
 String title=obj[3].toString();
 out.println("<tr>");
 out.println("<td bgcolor='#eeeeee'><div align='left'><font color='#eeeeee'><a href=view.jsp?ID="+id+">"+title+"</a></font></div></td>");
 out.println("<td width='15%' bgcolor='#ffff99'><div align='center'>"+name+"</div></td>");
 out.println("<td width='30%' bgcolor='#9999ff'><div align='center'>"+time+"</div></td>");
 out.println("</tr>");
}
%>
</table>
<div align="right">
<%
String fileName="page";//**************************将文件名作为变量***********************
out.print("第<font color=red>"+pageNum+"</font>页     共"+maxPageCount+"页&nbsp;&nbsp;&nbsp;");
if (maxPageCount>1)//不只有一页
{
  if (pageNum==1)//当前页是首页
   {
    out.print("首页 | 上一页 | <a href="+fileName+".jsp?page="+(pageNum+1)+">下一页</a> | ");
    out.print("<a href="+fileName+".jsp?page="+maxPageCount+">尾页</a> ");
    }//if (pageNum==1)
  else
   {
     if (pageNum==maxPageCount)//当前页是尾页
      {
       out.print("<a href="+fileName+".jsp?page=1>首页</a> | ");
       out.print("<a href="+fileName+".jsp?page="+(pageNum-1)+">上一页</a> | 下一页 | 尾页 ");
       }//if (pageNum==maxPageCount)
     else//当前页不是上面的2种情况
      {
       out.print("<a href="+fileName+".jsp?page=1>首页</a> | ");
       out.print("<a href="+fileName+".jsp?page="+(pageNum-1)+">上一页</a> | ");
       out.print("<a href="+fileName+".jsp?page="+(pageNum+1)+">下一页</a> | ");
       out.print("<a href="+fileName+".jsp?page="+maxPageCount+">尾页</a> ");
        }
   }
%>
跳转到第
  <select name="jumps" onChange="jumping(this)">
  <%
   for (int i=1;i<=maxPageCount;i++)
   {
     if (i==pageNum)
      {%>
    <option value="<%=fileName%>.jsp?page=<%=i%>" selected><%=i%></option>
    <%} else {%>
    <option value="<%=fileName%>.jsp?page=<%=i%>"><%=i%></option>
    <%} } %>
  </select>

<%
}//if (maxPageCount!=1)
%>
</div>
</body>
</html>
----------------------------------------------------------------------------------

    整个的分页功能写完后,我发现还是有很大的不足:显示分页的页面(page.jsp)代码太多,其他页面引用该功能的时候还是要从这里复制大段的代码,而且其中需要根据实际情况改动的地方多达七处(标注很多*的地方),这很容易出错,不利于管理使用,但是至今我仍找不到一个合适的方法解决这个问题——希望高手们扔玉。(这种写法确实很烂,但是不得不承认这是一个学习的过程,当你对自己的代码感到不满意并不断试图想要给自己一个满意的答卷时,你的水平就进步了)


原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 淘宝遇到职业敲诈怎么办 三无燕窝被打假怎么办 碰上职业打假的怎么办 遇到专业打假的怎么办 手机wifi被劫持怎么办 手机网页乱跳怎么办 老公说话不算话怎么办 编曲接不到活怎么办 电表显示跳闸黄灯怎么办 硫酸铬钾中毒怎么办 门锁能扭动却打不开怎么办 防盗门保险坏了怎么办 厦门工会卡过期怎么办 阴雨天被子受潮怎么办 衣服晾臭了怎么办 喷砂机油水分离泵有油怎么办 水太烫玻璃瓶盖打不开怎么办 玻璃罐的玻璃盖打不开怎么办? 开红酒没有开瓶器怎么办 手机卡突然坏了怎么办 滚筒洗衣机打不开门怎么办 全自动洗衣机门开不开怎么办 好期待评分不足怎么办 单片机数码管亮度低怎么办 猫的同轴灯不亮怎么办 楼下邻居太吵怎么办 冰箱冷却液内露怎么办 冷却水没有了会怎么办 金立m7信号不好怎么办 csgo凉了饰品怎么办 模型拟合度低怎么办 石膏线用发泡胶怎么办 电表上显示err10怎么办 电表显示欠压怎么办 tcl电视遥控器失灵怎么办 玩具汽车遥控器失灵怎么办 玩具车遥控丢了怎么办 按压扣坏了怎么办 电脑用不了鼠标怎么办 多肉上的肉虫子怎么办 警察被取消警衔怎么办