目录
一.重构-提取公用方法
1.为了进行公共方法的抽取,需要找出上面实习中的可通用部分,和差异化部分
2.公用方法封装思路
3. 具体实现
二.分页标签
2.1 准备一个Servlet
3.2 结果展示页面
三. 过滤器解决中文乱码问题
四.加入分页功能
四.封装分页标签
编写助手类
标签库描述文件中添加paging标签
使用分页标签首先在页面中引入标签
将原来的分页功能,替换为标签即可
MySQL分页
最后结果展示页面
将可通用的部分封装到模板中
差异化部分(即不可通用部分),可以定义一个处理接口,以便于通过参数传入个性化的实现部分
通用分页查询模板类:
public final class DBTemplate {
private DBTemplate() {
}
public static interface IORMConvert {
List convert(ResultSet rs) throws SQLException;
}
public static List query(String sql,
Object[] params,
PageBean pageBean,
IORMConvert convert) {
List datas = new ArrayList<>();
Connection con = null;
PreparedStatement ps = null;
ResultSet rs = null;
//不需要分页
if (pageBean == null || !pageBean.isPagination()) {
try {
con = DBUtil.getConection();
ps = con.prepareStatement(sql);
setParam(params, ps);
rs = ps.executeQuery();
datas = convert.convert(rs);
return datas;
} catch(Exception e) {
e.printStackTrace();
} finally {
DBUtil.closeDB(rs, ps, con);
}
} else {
//1. 查询总记录数
//2. 查询当前页数据
//1. 生成统计总记录数的SQL, 查询总记录数
try {
String countSql = "select count(*) from (" + sql + ") tmp";
con = DBUtil.getConection();
ps = con.prepareStatement(countSql);
setParam(params, ps);
rs = ps.executeQuery();
while(rs.next()) {
pageBean.setTotal(rs.getInt(1));
}
/*
* 如果统计的总记录数为0,则表示没有符合条件的记录,直接返回一个空结果集即可。
*/
if(pageBean.getTotal() == 0) {
return datas;
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if(pageBean.getTotal() == 0) {
DBUtil.closeDB(rs, ps, con);
}
DBUtil.closeDB(rs, ps);
}
//查询当前页数据
try {
String pagingSql = sql
+ " limit "
+ pageBean.getStartRow()
+ ", "
+ pageBean.getRows();
ps = con.prepareStatement(pagingSql);
setParam(params, ps);
rs = ps.executeQuery();
datas = convert.convert(rs);
} catch (SQLException e) {
e.printStackTrace();
} finally {
DBUtil.closeDB(rs, ps, con);
}
}
return datas;
}
private static void setParam(Object[] params, PreparedStatement ps) throws SQLException {
if (params != null) {
int i = 1;
for (Object param : params) {
ps.setObject(i, param);
i++;
}
}
}
}
使用示例:
public class StudentDao2 {
public List getStudents(String sname, PageBean pageBean) {
String sql = "select * from t_student where sname like ?";
return DaoTemplate.query(sql, new Object[] {sname}, pageBean, new IORMConvert() {
@Override
public List convert(ResultSet rs) throws SQLException {
List stus = new ArrayList<>();
while(rs.next()) {
Student stu = new Student();
stu.setSid(rs.getInt("sid"));
stu.setSname(rs.getString("sname"));
stu.setAge(rs.getInt("age"));
stu.setRemark(rs.getString("remark"));
stus.add(stu);
}
return stus;
}
});
}
public static void main(String[] args) {
StudentDao2 dao = new StudentDao2();
PageBean pageBean = new PageBean();
pageBean.setPage(3);
List students = dao.getStudents("张%", pageBean);
students.forEach(s -> System.out.println(s));
}
}
准备一个servlet用于处理请求,获取数据库中的数据,并转发到结果显示页面
@WebServlet(value = "/students")
public class StudentAction extends HttpServlet {
private static final long serialVersionUID = 3152900867611381148L;
private StudentDao2 studentDao = new StudentDao2();
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doPost(request, response);
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
PageBean pageBean = new PageBean();
pageBean.setRequest(request);
request.setAttribute("pageBean", pageBean);
String sname = request.getParameter("sname");
List students = studentDao.getStudents(sname+"%", pageBean);
request.setAttribute("students", students);
System.out.println("dopost .......... ");
request.getRequestDispatcher("/students/stuList.jsp").forward(request, response);
}
}
创建一个页面,该页面用于显示结果, 使用jstl的c标签来展示结果,为正常使用c标签,需要引入jstl-1.2.jar和standard-1.1.2.jar。
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"%>
<%@taglib prefix="z" uri="/zking" %>
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
Insert title here
学生信息
学号
姓名
年龄
备注
${student.sid}
${student.sname}
${student.age}
${student.remark}
/**
* 中文乱码处理
*/
@WebFilter("/*")
public class EncodingFiter implements Filter {
private String encoding = "UTF-8";// 默认字符集
public EncodingFiter() {
super();
}
public void destroy() {
}
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
HttpServletRequest req = (HttpServletRequest) request;
HttpServletResponse res = (HttpServletResponse) response;
// 中文处理必须放到 chain.doFilter(request, response)方法前面
res.setContentType("text/html;charset=" + this.encoding);
if (req.getMethod().equalsIgnoreCase("post")) {
req.setCharacterEncoding(this.encoding);
} else {
Map map = req.getParameterMap();// 保存所有参数名=参数值(数组)的Map集合
Set set = map.keySet();// 取出所有参数名
Iterator it = set.iterator();
while (it.hasNext()) {
String name = (String) it.next();
String[] values = (String[]) map.get(name);// 取出参数值[注:参数值为一个数组]
for (int i = 0; i < values.length; i++) {
values[i] = new String(values[i].getBytes("ISO-8859-1"),
this.encoding);
}
}
}
chain.doFilter(request, response);
}
public void init(FilterConfig filterConfig) throws ServletException {
String s = filterConfig.getInitParameter("encoding");// 读取web.xml文件中配置的字符集
if (null != s && !s.trim().equals("")) {
this.encoding = s.trim();
}
}
}
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8"%>
<%@taglib prefix="z" uri="/zking" %>
<%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
Insert title here
学生信息
学号
姓名
年龄
备注
${student.sid}
${student.sname}
${student.age}
${student.remark}
目前为止,分页功能已经实现了。
遗留下来的问题:
1 如果其他功能需要分页,则需要复制大量代码才能重用该功能
2 如果系统需要修改分页工具栏的显示风格呢?
为了方便代码的复用,及可维护性,我们将分页功能封装了一个自定义标签(其实就是将原来写在页面中的代码,通过移入到自定义标签中去实现),开发自定义标签分成三步:
public class PagingTag extends BodyTagSupport {
private PageBean pageBean;
public PageBean getPageBean() {
return pageBean;
}
public void setPageBean(PageBean pageBean) {
this.pageBean = pageBean;
}
@Override
public int doStartTag() throws JspException {
JspWriter out = this.pageContext.getOut();
try {
out.println(buildHtml());
return SKIP_BODY;
} catch (IOException e) {
throw new JspException("分页标签异常", e);
}
}
//生成Html内容
private String buildHtml() {
//构建分页页面元素
String pagingElement = "";
//构建隐藏表单,用于在分页时传递分页参数
String hiddenForm = "";
//构建分页功能需要的js代码块
String script = "";
return pagingElement + hiddenForm + script;
}
}
paging
com.zking.mvc.tag.PagingTag
empty
pageBean
true
true
<%@taglib prefix="z" uri="/zking" %>
limit语法
select * from table_name limit [offset,] rows
参数说明
参数 | 说明 |
---|---|
offset | 指定第一个返回记录行的偏移量(即从哪一行开始返回),注意:初始行的偏移量为0 |
offset | 返回具体行数 |