- 系统实现机制
- 涉及的关键技术
1. 系统实现机制
采集系统通过Strus2实现了MVC模式。
实现机制如下:
2. 涉及的关键技术
2.1 分页功能
分页功能涉及以下技术点:
- 分页需要的参数考虑
- 通过SQL语句实现高效分页
第一个技术点 分页需要考虑的参数
要实现分页,有四个参数是必须要考虑的:
- 当前页(current page):需要实时记录和传递,初始值为1
- 总的信息条数(all info count):通过数据库查询获取
- 每个页面要显示的信息数(page size):可以人为设置
- 总页数(total page) = all info count / page size
在实际的例子中,是写了一个PageRoll的类,将上面的四个参数都封装在这个类中。通过对于该类的访问,获取分页所需的数据。
第二个技术点 通过SQL语句实现高效分页
这里主要是通过SQL中ROW_NUMBER()语句实现分页,语句如下:
select temp.rownumber,* from
(select ROW_NUMBER() over (order by score desc) rownumber,* from KeyWords)
temp where temp.rownumber>? and temp.rownumber<=?";
// 填充sql语句
ps.setInt(1, (pageRoll.getCurrPage() - 1) * pageRoll.getPageSize());
ps.setInt(2, pageRoll.getCurrPage() * pageRoll.getPageSize());
先看子查询,ROW_NUMBER() over()函数首先是生成一个新的列,列名称是rownumber,而这一列是按照KeyWords表中的score字段降序排列的规则生成。有了这个新列(根据需求生成),就可以通过外层查询控制每次查询该列的消息数,进而控制了前台页面消息数的显示。
通过上述两个技术点,就能够实现分页了。
2.2 翻页功能实现
在讲这个知识点之前,也必须要弄清几个技术点:
- JSP——include的使用
- JSP动作元素<jsp:include>和<jsp:param>的搭配使用
- JSP中request与requestScope的区别
第一个技术点 JSP——include的使用
JSP中,include是一个经常用到的标签。当应用程序中所有的页面的某些部分(如标题、页脚和导航栏)都相同的时候,我们就可以考虑用include。但是相同的部分有静态的(装载进页面显示后再也不变的),有动态的(装载进页面后还会改变,如:随时间改变、随用户行为改变等)。
那么,对这两种类型的内容的包含方式一样吗?当然不一样。
第一种,静态包含 <%@ include file=”包含页面”%>
静态包含一般用于加载进页面显示后就再也不变的东西,比如页眉、背景、标题等等。静态包含不会检查所含文件的变化,把文件包含进来后,被包含文件的修改变化是不会影响已被包含进来的内容的。因为,静态包含发生在编译阶段。
比如:a.jsp中使用了语句 <%@ include file="b.jsp"%>,把b.jsp包含了进来,那么在编译a.jsp文件时,会直接把b.jsp文件的内容全部内嵌到a.jsp文件中包含b的语句的位置。然后运行a,显示a页面。也就是说,静态include是先把被包含文件的内容全部复制内嵌到包含文件中,再进行编译运行的。也正是因为要把b包含进a,所以b中的变量等不能与a重复,否则会报错。
第二种,动态包含 <jsp:include page=" " flush="true"/>
动态包含用于加载经常变化的、要求显示最新版本内容的东西,比如提交时间戳:用户打开博客编辑页面时,有一个时间加载进来了。用户编写完博客,点击提交时,就应该使用/显示提交瞬间的时间而不是打开编辑页面时的那个时间。所以这里要用的就是最新时间。
由上面我们知道,静态include是先包含进来,再编译,运行并传回浏览器显示的,所以不能满足我们要求某些部分使用最新内容的要求。那么,我们就要用到动态include。
动态include与静态include的最大不同在于:包含文件与被包含文件都是先编译执行,再包含。二者的编译阶段是相互独立的,只有在包含文件的include语句处把被包含文件的执行结果包含进来。换言之,包含文件先编译,执行。执行到了include语句的时候才触发被包含文件的编译、执行,并实时把结果包含进来。从而达到获取最新的被包含内容的目的。
同样使用a.jsp包含b.jsp的例子:加入a.jsp中动态include了b.jsp。现在,a先编译成servlet类文件,然后运行,当运行到包含b的语句处,引起b的编译,运行,并把b的运行servlet运行结果包含进a。最后a顺利运行完毕,把a的servlet类运行结果输出到浏览器显示。
第二个技术点 JSP动作元素<jsp:include>和<jsp:param>的搭配使用
当<jsp:include>和<jsp:param>动作元素一起使用时,可以将<jsp:param>中提供的参数值传递到<jsp:include>要加载的文件中去,因此当<jsp:include>和<jsp:param>结合使用时,可以在加载文件(页面)的过程中同时向该文件(页面)提供信息。
第三个技术点 JSP中request与requestScope的区别
(1)request对象通常用来接收从客户端通过表单提交过来的数据,然后在servlet或者action中用request.getParameter()的方法获取获取参数内容;
(2)requestScope通常是在servlet和action中通过request.setAttribute()方法把数据放到request对象中供客户端获取,然后客户端获取的方法就是requestScope.getAttribute()。
结论就是:
①request.getParameter(“username”)等价于{requestScope.username},一般是从服务器传递数据到页面,在页面中获取服务器保存在其中的数据内容。
OK,三个关键的技术点已经讲解清楚,下面进入正题:
实例代码如下:
后台显示页面(keyWords.jsp)include翻页页面(pageroll.jsp)的代码
<jsp:include page="../commonUser/pageroll.jsp" flush="true">
<jsp:param value="keywordsInback" name="action" />
</jsp:include>
上面代码中,设置flush为true,表明若缓冲区的内容很多了,就将数据读出,以免数据泄漏,造成错误。而keywordsInback是后台的一个方法,将其作为参数动态传递给pageroll.jsp页面。
当后台显示页面(keyWords.jsp)加载,数据渲染到此处时,首先将keywordsInback方法作为参数传递给pageroll.jsp。
pageroll.jsp,显示翻页的代码
<form action="${param.action}" name="pageRollForm" method="post">
当前是第${requestScope.pageRoll.currPage }页|
共${requestScope.pageRoll.pageCount }页|
共${requestScope.pageRoll.totalCount }条记录<br></br>
</form>
代码中,{requestScope.pageRoll.currPage }等价于request.getAttribute(“pageRoll.currPage”)获取当前页。
而通过${param.action},可以调用后台的keywordsInback方法,每次点击翻页时,都调用该方法,提交上面代码表单中的数据,获取到当前页数据,实现页面翻页显示。
keywordsInback方法,负责分页参数的获取、传递和处理
ArrayList<KeyWord> listData = keywordDao.getKeyWordsDBInbackground(pageRoll);//get data in database.
request.setAttribute("pageRoll", pageRoll);//set current page
request.setAttribute("list", listData);//set list data
上面代码负责返回分页所需数据(pageRoll)和后台页面渲染所需数据(listData)。注意,因为在pageroll.jsp中是通过${requestScope.pageRoll.currPage }来获取当前页的,所以此处必须要使用request.setAttribute,而不是request.setParameter。
至此,翻页功能就实现了。
ps:写文章真是件辛苦的差事,鬼知道我经历了什么!O(∩_∩)O