Java中JDBC實(shí)現(xiàn)動(dòng)態(tài)查詢(xún)的實(shí)例詳解
一 概述
1.什么是動(dòng)態(tài)查詢(xún)?
從多個(gè)查詢(xún)條件中隨機(jī)選擇若干個(gè)組合成一個(gè)DQL語(yǔ)句進(jìn)行查詢(xún),這一過(guò)程叫做動(dòng)態(tài)查詢(xún)。
2.動(dòng)態(tài)查詢(xún)的難點(diǎn)
可供選擇的查詢(xún)條件多,組合情況多,難以一一列舉。
3.最終查詢(xún)語(yǔ)句的構(gòu)成
一旦用戶向查詢(xún)條件中輸入數(shù)據(jù),該查詢(xún)條件就成為最終條件的一部分。
二 基本原理
1.SQL基本框架
無(wú)論查詢(xún)條件如何,查詢(xún)字段與數(shù)據(jù)庫(kù)是固定不變的,這些固定不變的內(nèi)容構(gòu)成SQL語(yǔ)句的基本框架,如
select column... from table。
2.StringBuilder形成DQL
獲取表單輸入,如果請(qǐng)求參數(shù)非空,根據(jù)該請(qǐng)求參數(shù)生成查詢(xún)條件,如“name=?”,“age>?”,將查詢(xún)條件追加到基本框架中。利用StringBuilder來(lái)追加查詢(xún)條件,這時(shí)出現(xiàn)一個(gè)問(wèn)題,怎么判斷生成的查詢(xún)條件中是否需要添加“and”?
如果該查詢(xún)條件是第一個(gè)查詢(xún)條件,不需要添加"and",否則需要添加“and”。問(wèn)題變得復(fù)雜起來(lái),每一次生成查詢(xún)條件時(shí)都需要判斷前面是否存在查詢(xún)條件。
我們可以考慮在SQL基本框架中添加一個(gè)查詢(xún)條件,該查詢(xún)條件的存在不影響查詢(xún)結(jié)果,只充當(dāng)占位角色,避免動(dòng)態(tài)添加查詢(xún)條件時(shí)判斷是否需要添加“and”。根據(jù)這些要求,這一查詢(xún)條件必須恒為真,這里我們?nèi) ?=1”,SQL基本框架就變成了
select column...from table where 1=1
每一個(gè)動(dòng)態(tài)查詢(xún)條件前段都添加“and”。
3.List集合為占位符賦值
有了DQL語(yǔ)句,接著需要考慮怎么為占位符賦值。可以在生成查詢(xún)條件的同時(shí),將占位符對(duì)應(yīng)的參數(shù)收集起來(lái),存入一個(gè)有序集合中,這里選擇List集合,這樣占位符就與List集合中的元素形成了順序上的對(duì)應(yīng)關(guān)系,第n個(gè)占位符對(duì)應(yīng)第n個(gè)元素,遍歷集合就可以為占位符賦值了。
為占位符賦值時(shí),不僅僅需要將數(shù)據(jù)傳遞給占位符,還需要選擇與字段一致的數(shù)據(jù)類(lèi)型,List集合僅僅存儲(chǔ)數(shù)據(jù)已經(jīng)不能夠滿足要求了,還需要添加字段信息,以區(qū)分不同的字段,選擇不同的數(shù)據(jù)類(lèi)型。這里集合中的元素采用“column+data”的形式。
三 Demo
1.數(shù)據(jù)庫(kù)
2.頁(yè)面
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<style>
span {
display: inline-block;
width: 75px;
margin-bottom: 15px;
}
</style>
<title>動(dòng)態(tài)查詢(xún)</title>
</head>
<body>
<form action="http://localhost:8080/JavaSETest/dynamicQueryServlet">
<div>
<span>姓名:</span><input type="text" name="name">
</div>
<div>
<span>性別:</span><input type="text" name="sex">
</div>
<div>
<span>年齡:</span><input type="text" name="age">
</div>
<div>
<span>部門(mén)編號(hào):</span><input type="text" name="depNo">
</div>
<div>
<input type="submit"value="查詢(xún)"> <input type="reset"value="重置">
</div>
</form>
</body>
</html>
3.服務(wù)器端(Servlet)
package com.javase.jdbc;
import java.io.IOException;
import java.io.PrintWriter;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/dynamicQueryServlet")
public class DynamicQueryServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=UTF-8");
// 獲取請(qǐng)求參數(shù)
String name = request.getParameter("name");
String sex = request.getParameter("sex");
String age = request.getParameter("age");
String depNo = request.getParameter("depNo");
// 關(guān)鍵是"where 1=1",不需要再判斷追加的查詢(xún)條件前是否需要添加and,統(tǒng)一在前面添加and
String baseSQL = "select name,sex,age,depNo from tb_employee where 1=1";
StringBuilder builder = new StringBuilder();// 用于拼接SQL語(yǔ)句
// 用于在占位符與參數(shù)值之間建立映射,占位符與參數(shù)值在各自序列中的排序一相同,例如name的占位符在SQL語(yǔ)句中排第一,name的參數(shù)值在
// 集合中排第一。
List<String> params = new ArrayList<String>();
builder.append(baseSQL);
if (isNotEmpty(name)) {
builder.append(" and name=? ");
params.add("name," + name);// 集合中不能僅僅存儲(chǔ)具體的數(shù)據(jù),還要存儲(chǔ)字段名,以便后續(xù)根據(jù)字段名選擇數(shù)據(jù)類(lèi)型
}
if (isNotEmpty(sex)) {
builder.append(" and sex=? ");
params.add("sex," + sex);
}
if (isNotEmpty(age)) {
builder.append(" and age=? ");
params.add("age," + age);
}
if (isNotEmpty(depNo)) {
builder.append(" and depNo=?");
params.add("depNo," + depNo);
}
Connection conn = null;
PreparedStatement ps = null;
ResultSet res = null;
StringBuilder resStr = new StringBuilder();
try {
conn = getConnection();
ps = conn.prepareStatement(builder.toString());
for (int i = 0; i < params.size(); i++) {
String str = params.get(i);
String[] arr = str.split(",");//arr[0]儲(chǔ)存字段信息,用于區(qū)分字段;arr[1]存儲(chǔ)數(shù)據(jù),用于為占位符賦值
// 因?yàn)闉檎嘉环x值時(shí),需要根據(jù)字段類(lèi)型選擇數(shù)據(jù)類(lèi)型,所以在此判斷類(lèi)型
if (arr[0].equals("age")) {
int a = Integer.parseInt(arr[1]);
ps.setInt(i + 1, a);
} else {
ps.setString(i + 1, arr[1]);
}
}
res = ps.executeQuery();
while (res.next()) {
String targetName = res.getString("name");
String targetSex = res.getString("sex");
int targetAge = res.getInt("age");
String targetDepNo = res.getString("depNo");
String temp = "name=" + targetName + "--" + "sex=" + targetSex + "--" + "age=" + targetAge + "--"
+ "depNo=" + targetDepNo;
resStr.append(temp + "<br>");
}
} catch (ClassNotFoundException | SQLException e) {
e.printStackTrace();
} finally {
if (res != null)
try {
res.close();
} catch (SQLException e) {
e.printStackTrace();
}
if (ps != null)
try {
ps.close();
} catch (SQLException e) {
e.printStackTrace();
}
if (conn != null)
try {
conn.close();
} catch (SQLException e) {
e.printStackTrace();
}
}
PrintWriter out = response.getWriter();
int length = resStr.length();
if (length == 0)
out.write("查詢(xún)?yōu)榭?);
else
out.write(builder.toString() + "<br>" + resStr.toString());
}
/**
* 判斷請(qǐng)求參數(shù)是否存在,是否有數(shù)據(jù)輸入
*
* @param str
* @return
*/
private boolean isNotEmpty(String str) {
if (str == null | str.equals("")) {
return false;
}
return true;
}
public static Connection getConnection() throws ClassNotFoundException, SQLException {
Class.forName("com.mysql.jdbc.Driver");
return DriverManager.getConnection("jdbc:mysql://localhost:3366/test01", "root", "123");
}
}
以上所述是小編給大家介紹的Java中JDBC實(shí)現(xiàn)動(dòng)態(tài)查詢(xún)的實(shí)例詳解,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- Java基礎(chǔ)開(kāi)發(fā)之JDBC操作數(shù)據(jù)庫(kù)增刪改查,分頁(yè)查詢(xún)實(shí)例詳解
- Java使用Jdbc連接Oracle執(zhí)行簡(jiǎn)單查詢(xún)操作示例
- JDBC連接MySql數(shù)據(jù)庫(kù)步驟 以及查詢(xún)、插入、刪除、更新等
- 使用jdbcTemplate查詢(xún)返回自定義對(duì)象集合代碼示例
- 使用JDBC從數(shù)據(jù)庫(kù)中查詢(xún)數(shù)據(jù)的方法
- JDBC使用游標(biāo)實(shí)現(xiàn)分頁(yè)查詢(xún)的方法
- java實(shí)現(xiàn)jdbc查詢(xún)結(jié)果集result轉(zhuǎn)換成對(duì)應(yīng)list集合
- JDBC查詢(xún)Map轉(zhuǎn)對(duì)象實(shí)現(xiàn)過(guò)程詳解
相關(guān)文章
JAVA實(shí)現(xiàn)基于Tcp協(xié)議的簡(jiǎn)單Socket通信實(shí)例
本篇文章主要介紹了JAVA實(shí)現(xiàn)基于Tcp協(xié)議的簡(jiǎn)單Socket通信實(shí)例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-01-01
關(guān)于Spring啟動(dòng)時(shí)Context加載源碼分析
這篇文章通過(guò)源碼分析主要給大家介紹了關(guān)于Spring啟動(dòng)時(shí)Context加載的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2018-01-01
淺談Java內(nèi)存模型之happens-before
于存在線程本地內(nèi)存和主內(nèi)存的原因,再加上重排序,會(huì)導(dǎo)致多線程環(huán)境下存在可見(jiàn)性的問(wèn)題。那么我們正確使用同步、鎖的情況下,線程A修改了變量a何時(shí)對(duì)線程B可見(jiàn)?下面小編來(lái)簡(jiǎn)單介紹下2019-05-05
java Spring AOP詳解及簡(jiǎn)單實(shí)例
這篇文章主要介紹了java Spring AOP詳解及簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-05-05
java設(shè)計(jì)模式之適配器模式(Adapter)
這篇文章主要介紹了java設(shè)計(jì)模式之適配器模式Adapter的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01

