springmvc+jpa实现分页的两种方式
1.工具类
public final class QueryTool {
public static PageRequest buildPageRequest(int pageNumber, int pageSize, String sortType){
Sort sort = null;
if("auto".equals(sortType)) {
sort = new Sort(Direction.DESC, "ctime");
} else {
sort = new Sort(Direction.DESC, "ctime");
}
return new PageRequest(pageNumber, pageSize, sort);
}
public static <T> Specification<T> buildSpecification(Map<String, Object> searchParams, Class<T> domainClass){
Map<String, SearchFilter> filters = SearchFilter.parse(searchParams);
Specification<T> spec = DynamicSpecifications.bySearchFilter(filters.values(), domainClass);
return spec;
}
}
2.Service类(只给出CommentService)
@Service
public class CommentService {
@Autowired
private CommentDao commentDao; @Transactional
public void addComment(Comment comment) {
if(comment == null) return;
commentDao.save(comment);
} @Transactional
public Page<Comment> findAllComments(String problemId, PageRequest pageable){
Page<Comment> page = commentDao.findAllCommentsByProblemId(problemId, pageable);
return page;
}
}
3.repository类(problem 和 comment 属于 一对多 的关系)
@Repository
public interface CommentDao extends JpaSpecificationExecutor<Comment>,
PagingAndSortingRepository<Comment, String> { @Query("select c from Comment c where c.problem.problemid = ?1")
public Page<Comment> findAllCommentsByProblemId(String problemId, Pageable pageable);
}
4.Controller类
第一种方式
@Controller
public class ProblemController { @Autowired
private ProblemService problemService; @Autowired
private CommentService commentService; @RequestMapping(value="getCurProblemComments")
@ResponseBody
public String getCurProblemComments(String problemName, @RequestParam(value = "pn", defaultValue = "1") int pageNumber,
@RequestParam(value = "ps", defaultValue = "4") int pageSize,
@RequestParam(value = "sortType", defaultValue = "auto") String sortType){
JSONObject jsono = new JSONObject();
try {
Problem problem = problemService.getProblemByName(problemName);
if(problem == null){
problem = problemService.addProblem(problemName);
}
PageRequest pageRequest = QueryTool.buildPageRequest(pageNumber, pageSize, sortType);
//根据题目的id获取所有的评论
Page<Comment> page = commentService.findAllComments(problem.getProblemid(), pageRequest);
//对 Comment 这个类中的一些字段进行过滤
SimplePropertyPreFilter spp = new SimplePropertyPreFilter();
spp.getExcludes().addAll(Comment.getExcludeString());
jsono.put("success", true);
jsono.put("message", JSON.toJSONString(page, spp));
} catch(Exception e){
e.printStackTrace();
jsono.put("success", false);
jsono.put("message", "inner error.");
}
return jsono.toString();
}
}
第二种方式,(懒了,有些内容是放在servic中的。)
通过JpaSpecificationExecutor<T> 的Page<T> findAll(Specification<T> spec, Pageable pageable); 方法(按照指定的规格条件)实现分页查询。
public String getCurProblemComments(@RequestParam(value = "pn", defaultValue = "1") int pageNumber,
@RequestParam(value = "ps", defaultValue = "2") int pageSize,
@RequestParam(value = "sortType", defaultValue = "auto") String sortType, ServletRequest request) {
Map<String, Object> searchParams = new HashMap<String, Object>();
searchParams = Servlets.getParametersStartingWith(request, "search_");
System.out.println(EncodeUtils.toUTF8((String)searchParams.get("LIKE_name")));
PageRequest pageRequest = QueryTool.buildPageRequest(pageNumber, pageSize, sortType);
searchParams.put(Operator.EQ + "_dr", "0");//逻辑删除中,字段dr=0表示这条数据没有删除
Specification<Comment> spec = QueryTool.buildSpecification(searchParams, Comment.class);
Page<Comment> page = commentDao.findAll(spec, pageRequest);
return JSON.toJSONString(page);
}
其中,Specification中的一些比较操作org.springside.modules.persistence.SearchFilter.Operator这个类中找到,如下:
public static enum Operator {
EQ, LIKE, GT, LT, GTE, LTE;
}
前台传递过来的参数名称中可以包含 Operator 中的一些操作, 比如参数名称是: search_LIKE_name (search表示要查询时比较, LIKE是比较的操作, name是表的字段名)。
5.java简单实现分页

分页面板类(MyPagePanel.java)
public class MyPagePanel extends JPanel{
private final int PAGE_BTN_NUM = 6;//按钮的数目的最大值, 一定为偶数
private JButton[] pageBtns = new JButton[PAGE_BTN_NUM];
private JButton preBtn = new JButton("<<");//前一页
private JButton nextBtn = new JButton(">>");//后一页
private JButton jumpBtn = new JButton("跳转");
private JTextField jumpText = new JTextField(5);//输入页号
private JLabel totPageLabel = new JLabel("共0页");//提示一共多少页
public static final int PAGE_SIZE = 2;
private CommunicationPanel parentPanel;
private ChatPanel chatPanel;
//刷新某页
public void refreshPage(){
if(curPage == -1){
JOptionPane.showMessageDialog(null, "请选择题目,然后再刷新!", "温馨提示",JOptionPane.WARNING_MESSAGE);
} else {
requestPage(curPage);
}
}
//请求某页
public void requestPage(int pageNumber){
//当前页按钮设为 true
if(curBtnPos >= 0)
pageBtns[curBtnPos].setEnabled(true);
JSONObject jsono = new JSONObject();
jsono.put("problemName", JavaRequest.problemName);
jsono.put("pageSize", String.valueOf(PAGE_SIZE));
jsono.put("pageNumber", String.valueOf(pageNumber));
String data = JavaRequest.sendPost("getCurProblemComments", jsono);
//获取当前题目的 评论记录
JSONObject result = JSONObject.fromObject(data);
if((Boolean) result.get("success")){//分页显示
JSONObject message = JSONObject.fromObject(result.get("message"));
Map<String, Class<?>> classMap = new HashMap<String, Class<?>>();
classMap.put("content", MyMessage.class);
MyPage myPage = (MyPage) JSONObject.toBean(message, MyPage.class, classMap);
this.updatePage(myPage);
} else {
JOptionPane.showMessageDialog(null, result.getString("message"), "温馨提示",JOptionPane.WARNING_MESSAGE);
if(result.get("returnLogin") != null && (Boolean)result.get("returnLogin")){//如果用户登录超时返回用户登陆界面
parentPanel.switchPanel(CommunicationPanel.LOGIN_PANEL);
}
}
}
//跳转都页面
private void jumpPage(int pageNumber) throws Exception{
if(pageNumber < 0 || pageNumber >= totalPages)
throw new Exception("不存在该页号!");
requestPage(pageNumber);
}
public MyPagePanel(CommunicationPanel parentPanelx, ChatPanel chatPanelx){
this.parentPanel = parentPanelx;
this.chatPanel = chatPanelx;
preBtn.setEnabled(false);
this.add(preBtn);
preBtn.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
requestPage(curPage-1);
}
});
for(int i=0; i<pageBtns.length; ++i) {
pageBtns[i] = new JButton();
this.add(pageBtns[i]);
pageBtns[i].setVisible(false);
pageBtns[i].addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
//JButton text 表示的是页号
String text = ((JButton)e.getSource()).getText();
int pageNumber = Integer.valueOf(text);
requestPage(pageNumber);
}
});
}
nextBtn.setEnabled(false);
this.add(nextBtn);
nextBtn.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
requestPage(curPage+1);
}
});
jumpBtn.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
try{
if(jumpText.getText().isEmpty()) throw new Exception("页数不能为空!");
int pageNumber = Integer.valueOf(jumpText.getText());
jumpPage(pageNumber);
} catch (Exception ex){
ex.printStackTrace();
JOptionPane.showMessageDialog(null, "请确定页号是否正确!", "温馨提示",JOptionPane.WARNING_MESSAGE);
}
}
});
this.add(totPageLabel);
this.add(jumpText);
this.add(jumpBtn);
}
private int curPage = -1, totalPages, curBtnPos = -1;
public void updatePage(MyPage myPage){
//清除当页面的信息
chatPanel.getJpChat().setText("");
//处理接受到的 评论消息
for(MyMessage message : myPage.getContent()){
chatPanel.addRecMsg(message.getCcontent(), StyleConstants.ALIGN_LEFT);
}
curPage = myPage.getNumber();
totalPages = myPage.getTotalPages();
totPageLabel.setText("共" + totalPages + "页");
if(curPage == 0) preBtn.setEnabled(false);
else preBtn.setEnabled(true);
if(curPage == totalPages-1 || totalPages == 0) nextBtn.setEnabled(false);
else nextBtn.setEnabled(true);
List<Integer> pageNum = new LinkedList<Integer>();
if(totalPages > 0) {
int step = PAGE_BTN_NUM/2-1;
for(int i=step; i>=1; --i) {
if(curPage-i>=0)
pageNum.add(curPage-i);
}
pageNum.add(curPage);
for(int i=1; i<=step+1 && curPage+i<totalPages; ++i)
pageNum.add(curPage+i);
if(pageNum.size() < PAGE_BTN_NUM){
while(pageNum.size() < PAGE_BTN_NUM && pageNum.get(0)-1 >= 0)
pageNum.add(0, pageNum.get(0)-1);
while(pageNum.size() < PAGE_BTN_NUM && pageNum.get(pageNum.size()-1)+1 < totalPages)
pageNum.add(pageNum.get(pageNum.size()-1)+1);
}
}
int bi = 0;
for(int pi=0; pi < pageNum.size(); ++pi, ++bi){
pageBtns[bi].setText(String.valueOf(pageNum.get(pi)));
if(pageNum.get(pi) == curPage){
curBtnPos = bi;//记录这个按钮的页号值 和 当前页号值相等
}
}
//重新显示按钮
for(int i=0; i<bi; ++i)
pageBtns[i].setVisible(true);
//后面这些按钮都是没有 用上的
for(int i=bi; i < pageBtns.length; ++i){
pageBtns[i].setVisible(false);
}
if(totalPages > 0)
pageBtns[curBtnPos].setEnabled(false);
}
}
MyMessage.class
public class MyMessage{
private String ccontent;
private String commentid;
public String getCcontent() {
return ccontent;
}
public void setCcontent(String ccontent) {
this.ccontent = ccontent;
}
public String getCommentid() {
return commentid;
}
public void setCommentid(String commentid) {
this.commentid = commentid;
}
}
MyPage.class
public class MyPage{
private List<MyMessage> content;
private boolean first;
private boolean last;
private int number;
private int numberOfElements;
private int size;
private String totalElements;
private String sort;
public String getSort() {
return sort;
}
public void setSort(String sort) {
this.sort = sort;
}
public List<MyMessage> getContent() {
return content;
}
public void setContent(List<MyMessage> content) {
this.content = content;
}
public boolean isFirst() {
return first;
}
public void setFirst(boolean first) {
this.first = first;
}
public boolean isLast() {
return last;
}
public void setLast(boolean last) {
this.last = last;
}
public int getNumber() {
return number;
}
public void setNumber(int number) {
this.number = number;
}
public int getNumberOfElements() {
return numberOfElements;
}
public void setNumberOfElements(int numberOfElements) {
this.numberOfElements = numberOfElements;
}
public int getSize() {
return size;
}
public void setSize(int size) {
this.size = size;
}
public String getTotalElements() {
return totalElements;
}
public void setTotalElements(String totalElements) {
this.totalElements = totalElements;
}
public int getTotalPages() {
return totalPages;
}
public void setTotalPages(int totalPages) {
this.totalPages = totalPages;
}
private int totalPages;
}
实现思路:
1. 首先看一下后台的json数据的格式(其中message的值是spring Data JPA分页查询结果的json)
{"success":true,"message":{"content":[{"ccontent":"hjz 2016-38-11 15:38:21*****宋体|12|0-0-0| *****0$$$$$70#####","commentid":"8ae45d92549da2a801549ec092ce001f"},{"ccontent":"hjz 2016-38-11 15:38:10*****宋体|12|0-0-0|哇哇 *****2$$$$$55#####","commentid":"8ae45d92549da2a801549ec068c7001e"}],"first":true,"last":false,"number":0,"numberOfElements":2,"size":2,"sort":{},"totalElements":8,"totalPages":4}}
2. 将“message”对应的值转换成MyPage对象,因为MyPage中有一个List<MyMessage>,如果属性中含有复杂的类型,当其中属性有类似List , Map ,ArrayList、自定义的类型,如List<MyMessage> content, 普通的转换会报错:java.lang.ClassCastException: net.sf.ezmorph.bean.MorphDynaBean cannot be cast to XXX,在JSONObject.toBean的时候如果转换的类中有集合,可以先定义Map<String, Class> classMap = new HashMap<String, Class>();在classMap中put你要转换的类中的集合名 。本例中需要将“content”对应的值转换成 MyMessage对象。
//data为后台的json串
JSONObject result = JSONObject.fromObject(data);
if((Boolean) result.get("success")){//分页显示
JSONObject message = JSONObject.fromObject(result.get("message"));
Map<String, Class<?>> classMap = new HashMap<String, Class<?>>();
classMap.put("content", MyMessage.class);
MyPage myPage = (MyPage) JSONObject.toBean(message, MyPage.class, classMap);
this.updatePage(myPage);
}
3.然后通过分页的信息,简单的实现分页按钮,详情看MyPagePanel.java
springmvc+jpa实现分页的两种方式的更多相关文章
- 【Spring】SpringMVC非注解配置的两种方式
目录结构: contents structure [+] SpringMVC是什么 Spring MVC的设计原理 SpringMVC配置的第一种方式 1,复制Jar包 2,Web.xml文件 3,M ...
- SpringMVC配置字符过滤器的两种方式
有时候使用SpringMVC框架提交表单时会出现中文乱码,以下是我亲自试验过的配置字符过滤器的两种: 1.在web.xml中配置 <filter> <filter-name>c ...
- springmvc获取资源文件的两种方式(超简单)
1 比如我们在sc目录下新建一个db.properties文件内容如下 DriverClass=com.mysql.jdbc.Driverurl=jdbc:mysql://localhost:3306 ...
- springmvc+druid+dataSource配置的两种方式
一.一般的配置方式 数据库连接配置在jdbc.properties文件中,这种方式有一个最大的缺点,数据库的配置信息对开发人员是完全可见的,十分方便程序员删库跑路.spring配置具体如下: 1.jd ...
- yii2-搜索带分页,分页的两种方式
1.文章表关联 <?php //...other code //关联 public function getCate(){ return $this->hasOne(ArticleCate ...
- django分页的两种方式
第一种自定义分页: def pageDemo(request): ''' 自定义分页] :param request: :return: ''' currentpage=request.GET.get ...
- ibatis分页的两种方式
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...
- oracle 分页的两种方式
实例:查询5-8名学生的姓名与成绩 --oracle的分页1 between 方式(分三次查询,第一次只作排序,第二次给表加上rownum序列,第三次为查询结果) select s.scorenumb ...
- mysql中用limit 进行分页有两种方式
代码示例:语句1: select * from student limit 9,4 语句2: slect * from student limit 4 offset 9 // 语句1和2均返回表stu ...
随机推荐
- Java MyBatis 插入数据库返回主键
最近在搞一个电商系统中由于业务需求,需要在插入一条产品信息后返回产品Id,刚开始遇到一些坑,这里做下笔记,以防今后忘记. 类似下面这段代码一样获取插入后的主键 User user = new User ...
- DDD CQRS架构和传统架构的优缺点比较
明天就是大年三十了,今天在家有空,想集中整理一下CQRS架构的特点以及相比传统架构的优缺点分析.先提前祝大家猴年新春快乐.万事如意.身体健康! 最近几年,在DDD的领域,我们经常会看到CQRS架构的概 ...
- 阿里签名中URLEncode于C#URLEncod不同之处
问题 如上图所示,阿里云的PercentEncode 转换! 为 %21 PercentEncode 源码为: package com.aliyuncs.auth; import java.io.Un ...
- android键盘
在应用的开发过程中有不少的情况下会用到自定义键盘,例如支付宝的支付密码的输入,以及类似的场景.android系统给开发者们提供了系统键盘,KeyboardView,其实并不复杂,只是有些开发者不知道罢 ...
- 6. ModelDriven拦截器、Preparable 拦截器
1. 问题 Struts2 的 Action 我们将它定义为一个控制器,但是由于在 Action 中也可以来编写一些业务逻辑,也有人会在 Action 输入业务逻辑层. 但是在企业开发中,我们一般会将 ...
- 阿里云服务器上配置并使用: PHP + Redis + Mysql 从配置到使用
(原创出处为本博客,http://www.cnblogs.com/linguanh/) 目录: 一,下载 二,解压 三,配置与启动 四,测试 Redis 五,配置 phpRedis 扩展 六,综合测试 ...
- AFNetworking 3.0 源码解读(十)之 UIActivityIndicatorView/UIRefreshControl/UIImageView + AFNetworking
我们应该看到过很多类似这样的例子:某个控件拥有加载网络图片的能力.但这究竟是怎么做到的呢?看完这篇文章就明白了. 前言 这篇我们会介绍 AFNetworking 中的3个UIKit中的分类.UIAct ...
- Android 微信第三方登录(个人笔记)
今天在写微信登录,花了半天时间搞定.然后写下自己的笔记,希望帮助更多的人...欢迎各位指教. 微信授权登录,官方说的不是很清楚.所以导致有一部分的坑. 微信注册应用平台的应用签名,下载 微信签名生成工 ...
- 前端开发小白必学技能—非关系数据库又像关系数据库的MongoDB快速入门命令(2)
今天给大家道个歉,没有及时更新MongoDB快速入门的下篇,最近有点小忙,在此向博友们致歉.下面我将简单地说一下mongdb的一些基本命令以及我们日常开发过程中的一些问题.mongodb可以为我们提供 ...
- kafka
2016-11-13 20:48:43 简单说明什么是kafka? Apache kafka是消息中间件的一种,我发现很多人不知道消息中间件是什么,在开始学习之前,我这边就先简单的解释一下什么是消息 ...