如何更準確過濾信息?看完本篇你就知道

不管是使用Excel辦公,仍是瀏覽器搜尋關鍵字,都是須要經過條件過濾來實現。今天,手把手教你實現Java web項目——實現多條件過濾功能。javascript

分頁查詢
需求分析:在列表頁面中,顯示指定條數的數據,經過翻頁按鈕完成首頁/上一頁/下一頁/尾頁的查詢
數據分析:
經過觀察,頁面上須要顯示下面的幾個數據:
當前頁:currentPage
頁面大小:pageSize
總頁數:totalPage
首頁:1
上一頁:prevPage
下一頁:nextPage
尾頁:endPage
總條數:totalCount
結果集:resultjava

如何更準確過濾信息?看完本篇你就知道br/>那麼,咱們應該如何方便快速的將這多個數據共享到頁面上呢?答案是:封裝
咱們應該將這幾個參數封裝到一個對象中,而後共享這個對象便可,因此,咱們有了下面這個類
@Getter
public class PageResult {
public static final PageResult EMPTY_RESULT = new PageResult(Collections.EMPTY_LIST, 0, 1, 3);
//1:兩傳
private int currentPage;
private int pageSize;web

//2:兩查
private List<?> result;
private int totalCount;

//3:三計算
private int prevPage;
private int nextPage;
private int endPage;

public PageResult(List<?> result, int totalCount, int currentPage, int pageSize){

    this.result = result;
    this.totalCount = totalCount;
    this.pageSize = pageSize;
    this.currentPage = currentPage;
    //計算
    this.endPage = totalCount % pageSize == 0 ?
                totalCount / pageSize : totalCount / pageSize  + 1; 
    this.prevPage = currentPage - 1 > 0 ? currentPage - 1 : 1;
    this.nextPage = currentPage + 1 > endPage ? endPage :currentPage + 1;
}

}
在這個類中,咱們提供了一個構造器來快速封裝數據
其中,endPage/prevPage/nextPage是經過上面的幾個參數計算得來的
在這些數據中,存在兩個須要從數據庫中查詢獲得的數據:總條數/結果集
這兩個數據咱們須要下面兩條SQL進行查詢
查詢部門表中數據的總條數
SELECT count(id) FROM department
使用LIMIT關鍵字查詢指定頁面的數據
SELECT id, name, sn FROM department LIMIT #{start}, #{pageSize}
#{start}: 使用(currentPage-1)pageSize表達式計算出來的開始索引
#{pageSize}: 每次查詢的最大條數br/>要執行這兩條SQL,須要用戶傳遞兩個參數:currentPage和pageSize
爲了參數方便傳遞,咱們將這兩個參數封裝到一個類中:QueryObject
@Setter
@Getter
public class QueryObject {
// 默認查詢第一頁的數據
private int currentPage = 1;
// 頁面中默認顯示10條數據
private int pageSize = 5;
public int getStart(){
return (currentPage - 1)
pageSize;
}
}
能夠看出,查詢結果集中的#{start}表達式,是訪問查詢對象中的getStart()方法來獲取到計算獲得的開始索引
到此,咱們都已經封裝好了分頁查詢中最核心的兩個類:
QueryObject:封裝用戶傳遞過來的currentPage/pageSize
PageResult:封裝頁面上顯示須要的result/totalCount/currentPage/pageSize/totalPage/prevPage/pageSize
有了這兩個類,咱們就能夠在service中定義下面的方法,來處理分頁查詢的業務了:sql

public PageResult query(QueryObject qo) {
    //查詢表中數據的總條數
    int totalCount = dao.queryForCount(qo);
    //當查詢到的總條數爲0時,說明沒有數據,此時就不該該再以後下面的查詢
    //直接返回相應的默認值便可
    if (totalCount == 0) {
        return PageResult.EMPTY_RESULT;
    }
    List<Department> data = dao.queryForList(qo);
    PageResult result = new PageResult(data, totalCount, qo.getCurrentPage(), qo.getPageSize());
    return result;
}

該方法接收用戶傳遞的數據(QueryObject),返回用戶須要的數據(PageResult)
經過調用dao中的兩個方法執行兩條SQL查詢數據(總條數和結果集)
<!--查詢總條數-->
<select id="queryForCount" resultType="java.lang.Integer">
SELECT count(id)
FROM department
</select>
<!--查詢結果集-->
<select id="queryForList" resultType="Department">
SELECT
id,
name,
sn
FROM department
LIMIT #{start}, #{pageSize}
</select>
而後將數據封裝到PageResult對象中返回給表現層
表現層獲取到service中封裝的PageResult對象後,共享到request做用域中
而後請求轉發到list.jsp頁面
protected void list(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//接收用戶傳遞的currentPage和pageSize
String currentPage = req.getParameter("currentPage");
String pageSize = req.getParameter("pageSize");
//將數據封裝到QueryObject中,傳遞給service進行處理
QueryObject qo = new QueryObject();
if(!StringUtils.isNullOrEmpty(currentPage)){
qo.setCurrentPage(Integer.valueOf(currentPage));
}
if(!StringUtils.isNullOrEmpty(pageSize)){
qo.setPageSize(Integer.valueOf(pageSize));
}
PageResult result = service.query(qo);
//共享獲取到的PageResult對象
req.setAttribute("result", result);
req.setAttribute("qo", qo);
//請求轉發回到list.jsp頁面
req.getRequestDispatcher("/WEB-INF/views/department/list.jsp").forward(req, resp);
}
在list.jsp頁面上,使用EL+JSTL獲取數據並顯示在對應的位置
部門列表
<c:forEach items="${result.data}" var="entity" varStatus="vs">
<tr>
<td>${vs.count}</td>
<td>${entity.name}</td>
<td>${entity.sn}</td>
<td>
<a class="btn btn-info btn-xs" href="/department?cmd=input&id=${entity.id}">
<span class="glyphicon glyphicon-pencil"></span>編輯
</a>
<a href="/department?cmd=delete&id=${entity.id}" class="btn btn-danger btn-xs">
<span class="glyphicon glyphicon-trash"></span>刪除
</a>
</td>
</tr>
</c:forEach>
分頁條
<div style="text-align: center;">
<a href="javascript:;" onclick="goPage(1)">首頁</a>
<a href="javascript:;" onclick="goPage(${result.prevPage})">上一頁</a>
<a href="javascript:;" onclick="goPage(${result.nextPage})">下一頁</a>
<a href="javascript:;" onclick="goPage(${result.endPage})">尾頁</a>數據庫

當前頁: ${result.currentPage} / ${result.endPage}

 跳轉到第: <input id="currentPage" name="currentPage" style="width: 80px; text-align: center;"
              type="number" min="1" max="${result.endPage}" value="${result.currentPage}"/> 頁
 <input type="submit" value="GO"/>
 每頁顯示:
 <select name="pageSize" onchange="goPage(1);">
     <option ${qo.pageSize==5?"selected='selected'":""}>5</option>
     <option ${qo.pageSize==10?"selected='selected'":""}>10</option>
     <option ${qo.pageSize==15?"selected='selected'":""}>15</option>
     <option ${qo.pageSize==20?"selected='selected'":""}>20</option>
 </select>
 條

</div>
效果以下
如何更準確過濾信息?看完本篇你就知道瀏覽器

說明:
在點擊翻頁的時候,經過執行相應的JS代碼提交表單來發起請求
將須要查詢的當前頁的值設值給表單中的id爲currentPage的輸入框,而後提交表單
目的主要是和後面的高級查詢進行合併使用
作到這裏,咱們部門的分頁功能就完成了
由於部門的字段比較少,因此,在這個模塊中,沒有設計高級查詢的功能,這個功能咱們在員工模塊中再去實現
接下來,咱們來看看員工模塊相應功能的實現mybatis

在完成部門的CRUD和分頁查詢後發現,其餘模塊的這些功能基本類似
不一樣之處主要在於字段不一樣而已,因此,在這裏,咱們主要對這些不一樣點進行說明,其餘的按照前面的實現便可
首先,來看看員工的表結構
如何更準確過濾信息?看完本篇你就知道app

image.png
在該表中,前六個字段都是基本的字段,第七個(dept_id),這個字段是關聯部門的外鍵列
因此,待會兒咱們在完成CRUD的過程當中,須要注意維護該字段的值
清楚表結構以後,咱們來對員工的CRUD作一個簡單的分析
查詢功能:
如何更準確過濾信息?看完本篇你就知道
能夠看到,處理部門這一列顯示的數據比較特殊以外,其餘的都是基本的數據展現而已
什麼是特殊?什麼是不特殊?
員工除了部門的信息外,其餘的數據都是直接來自於數據庫,而部門在表中只存儲了對應的編號,可是頁面上須要顯示部門的名稱,那麼這個問題咱們是怎麼解決的呢?
好,首先解釋一下,這個問題的解決方案在目前咱們的web階段尚未涉及過,因此,我在這隻能點到爲止
咱們的解決方案是:在執行該數據的查詢的時候,使用多表查詢,將員工及其所在部門的信息查詢出來,SQL以下:
SELECT
e.id,
e.name,
e.password,
e.email,
e.age,
e.admin,
e.dept_id,
d.id d_id,
d.name d_name,
d.sn d_sn
FROM employee e LEFT JOIN department d on e.dept_id = d.id
limit #{start}, #{pageSize}
這條SQL可以查詢到以下的結果jsp

如何更準確過濾信息?看完本篇你就知道
而後在resultMap中以下的配置,完成數據的封裝,員工相關的數據封裝到Employee對象中,部門相關的數據封裝到Department對象中
<resultMap id="BaseResultMap" type="Employee">
<id column="id" jdbcType="BIGINT" property="id"/>
<result column="name" jdbcType="VARCHAR" property="name"/>
<result column="password" jdbcType="VARCHAR" property="password"/>
<result column="email" jdbcType="VARCHAR" property="email"/>
<result column="age" jdbcType="INTEGER" property="age"/>
<result column="admin" jdbcType="BIT" property="admin"/>
<!-- 一對多關係 -->
<association property="dept" javaType="Department">
<id column="d_id" property="id"/>
<result column="d_name" property="name"/>
<result column="d_sn" property="sn"/>
</association>
</resultMap>
最後,在select元素中使用resultMap來完成映射
<select id="queryForList" resultMap="BaseResultMap">
完成這些操做以後,咱們獲取到的每一個員工及其所在的部門信息就封裝好了,那麼,在JSP頁面中使用EL表達式,就可以獲取到當前員工所在部門的相關信息了
<td>${entity.dept.name}</td>
查詢功能分析到此結束,其餘功能和部門的一致
新增功能:
需求分析:
保存用戶的相關信息,包括用戶所在部門的信息
經過對員工表的觀察發現,表中關聯了部門的主鍵信息,來講明當前員工所在的部門
因此,在保存員工的時候,須要爲員工設置所在的部門.頁面設計以下
如何更準確過濾信息?看完本篇你就知道ide

爲員工設置所在的部門,用戶只須要進行選擇便可,而後將選中部門的id傳遞到服務端
那麼問題來了,如何將部門id經過下拉框傳遞到服務端呢?請看下面的分析
protected void input(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
List<Department> departments = departmentService.list();
req.setAttribute("departments", departments);
String id = req.getParameter("id");
if (!StringUtils.isNullOrEmpty(id)) {
Employee employee = service.get(Long.valueOf(id));
req.setAttribute("employee", employee);
}
req.getRequestDispatcher("/WEB-INF/views/employee/input.jsp").forward(req, resp);
}
<select class="form-control" id="dept" name="deptId">
<c:forEach var="d" items="${departments}">
<option value="${d.id}" >${d.name}</option>
</c:forEach>
</select>
首先,在跳轉到該頁面之間,咱們先將全部的部門信息查詢到
而後在頁面上循環遍歷,生成對應的option元素,將部門的id做爲option的value(提交的數據),將部門的name做爲option的文本內容(顯示的數據)
最後,在用戶選擇對應選項後,提交表單,會將對應option元素的value提交到後臺,完整數據以下:
如何更準確過濾信息?看完本篇你就知道

經過上圖,能夠清晰的看到,咱們已經將完整的數據提交到後臺
後臺獲取到這些數據以後,就可以將其保存到數據庫中
那麼,咱們的保存功能就算完成了
更新功能:
更新和保存只有一個地方不一樣,就是須要數據回顯
而數據回顯中,咱們只對部門(下拉框)和超級管理員(複選框)作一個說明,其餘的由於都是普通的input元素,使用value屬性顯示數據便可
而下拉框和複選框須要單獨進行處理,以下:
<select class="form-control" id="dept" name="deptId">
<c:forEach var="d" items="${departments}">
<option value="${d.id}" ${employee.dept.id==d.id?"selected":""}>${d.name}</option>
</c:forEach>
</select>

<input type="checkbox" id="admin" name="admin" ${employee.admin?'checked':''}>
這裏,咱們選擇使用EL表達式的三元運算符進行判斷,爲下拉框添加selected屬性,爲複選框添加checked屬性
編輯的時候,數據能回顯,接下來的操做和新增一致
刪除功能
和部門的刪除一致,這裏就再也不贅述
到此,員工的CRUD結束
高級查詢
功能需求:
輸入關鍵字和部門信息進去過濾查詢,關鍵字是根據姓名和郵箱兩個字段查詢
頁面設計
如何更準確過濾信息?看完本篇你就知道

高級查詢效果圖
當用戶輸入關鍵字"趙"和部門"總經辦"後,在列表中顯示的查詢結果則爲全部總經辦姓趙的員工
那麼此時應該執行下面的SQL來查詢相應的數據
SELECT
e.id, e.name, e.password, e.email, e.age, e.admin, e.dept_id, d.id d_id, d.name d_name, d.sn d_sn
from
employee e
LEFT JOIN
department d
ON
e.dept_id = d.id
WHERE
(e.name LIKE concat('%',? ,'%') OR e.email LIKE concat('%',? ,'%')) AND e.dept_id = ? LIMIT ?, ? br/>在該SQL中,WHERE後面的條件是根據用戶傳遞的高級查詢相關的參數拼接而來
這裏,咱們須要解決兩個問題:
1.這裏的多個高級查詢的參數如何傳遞?
對於這個問題,咱們應該可以比較快的想到解決方案---數據封裝,以下:
@Setter
@Getter
public class EmployeeQueryObject extends QueryObject {
private String keyword; //按照員工名稱與郵箱模糊查詢
private Long deptId = -1L; //按照部門id查詢

//當keyword爲null或者空字符串的時候,都視爲null處理
public String getKeyword(){
    return StringUtils.isEmpty(keyword) ? null : keyword;
}

}
2.參數拿到後,如何拼接到對應的SQL中?
使用mybatis中的動態SQL中提供的標籤,在mapper映射文件中進行SQL的拼接
<sql id="base_where">
<where>
<if test="keyword != null">
AND( e.name LIKE concat('%',#{keyword} ,'%') OR e.email LIKE concat('%',#{keyword} ,'%'))
</if>
<if test="deptId > 0">
AND e.dept_id = #{deptId}
</if>
</where>
</sql>
以上兩個問題解決後,咱們就能夠根據用戶傳遞過來的參數,執行對應的過濾查詢的SQL
最後,和分頁查詢的邏輯同樣,將數據封裝到PageResult中,和分頁相關的數據一同返回到頁面
效果如[高級查詢效果圖]所示
好了,基本功能是完成了,咱們再來看看下面的問題:
首先,在表單中輸入下面的參數,而後查詢
如何更準確過濾信息?看完本篇你就知道

而後點擊下一頁進行翻頁
如何更準確過濾信息?看完本篇你就知道
能夠看到,在點擊翻頁以後,咱們不是在上面的基礎上查詢下一頁的數據,而是查詢到了全部的數據,WHY?
其實很簡單,來看看咱們的請求參數,一切就清楚了
在咱們點擊翻頁的時候,傳遞的參數只有currentPage,並無將以前的高級查詢的參數一塊兒傳遞到後臺,因此執行了下面的SQL查詢到上面的結果
SELECT
e.id, e.name, e.password, e.email, e.age, e.admin, e.dept_id, d.id d_id, d.name d_name, d.sn d_sn
from
employee e
LEFT JOIN
department d
on
e.dept_id = d.id limit ?, ?
因此,要想在以前高級查詢的基礎上,繼續進行分頁查詢,那麼咱們只有一個辦法,就是在翻頁的時候將高級查詢和分頁的參數一塊兒提交到後臺,拼接執行相應的SQL才行
解決方案:
使用JS來完成數據的提交(JS在目前階段還未進行系統學習,因此,這裏你們重點應該是放在咱們要作的事情上,而不是怎麼作)
<script type="text/javascript">
function goPage(currentPage) {
//爲表單中的currentPage輸入框設值
document.getElementById("currentPage").value = currentPage
//提交表單
document.forms[0].submit();
}
</script>

翻頁條
<a href="javascript:;" onclick="goPage(1)">首頁</a>
<a href="javascript:;" onclick="goPage(${result.prevPage})">上一頁</a
<a href="javascript:;" onclick="goPage(${result.nextPage})">下一頁</a
<a href="javascript:;" onclick="goPage(${result.endPage})">尾頁</a>
在點擊翻頁按鈕時,調用定義好的goPage函數,完成其中的兩件事便可
此時,咱們能夠看到問題已然解決
最後,頁面跳轉和頁面大小的設置功能,咱們不作要求,若是要實現也很簡單,我這裏就直接把相關代碼貼出來
跳轉到第: <input id="currentPage" name="currentPage" style="width: 80px; text-align: center;"
type="number" min="1" max="${result.endPage}" value="${result.currentPage}"/> 頁
<input type="submit" value="GO"/>
每頁顯示:
<select name="pageSize" onchange="goPage(1);">
<option ${qo.pageSize==5?"selected='selected'":""}>5</option>
<option ${qo.pageSize==10?"selected='selected'":""}>10</option>
<option ${qo.pageSize==15?"selected='selected'":""}>15</option>
<option ${qo.pageSize==20?"selected='selected'":""}>20</option>
</select>

好了,高級查詢的功能實現到此結束,咱們來作一個小結
高級查詢和分頁的功能,咱們應該重點分析兩個點
1.用戶須要看到什麼數據?
2.這些數據應該執行什麼樣的SQL才能查詢到?
若是將這兩個問題分析清楚了,那麼你們就可以知道這個過程當中所封裝的幾個類的做用了
QueryObject:封裝查詢對象中通用的屬性
EmployeeQueryObject:封裝高級查詢相關的條件參數
PageResult:封裝用戶須要使用到的數據
因此實現步驟大體以下:
1.獲取到用戶傳遞的高級查詢和分頁的參數,封裝到對象的查詢對象中
2.從查詢對象中取出數據,而後拼接SQL
3.將查詢以後獲得的結果,封裝到PageResult對象中
4.頁面獲取到PageResult中的數據進行顯示
點擊查看JavaWeb系列的其它文章
手把手教你作JavaWeb項目:項目需求分析
手把手教你作JavaWeb項目:前臺界面
手把手教你作JavaWeb項目:登陸模塊

多條件過濾確實給咱們帶來了不少便利,咱們coder確實須要去掌握這樣的方法,才能讓本身的項目更佳完善。最後呢,但願各位coder可以從分享中獲得本身想要的,亦或者是一些啓發或者共鳴。

相關文章
相關標籤/搜索