JDBC實現動態查詢

来源:http://www.cnblogs.com/tonghun/archive/2017/07/07/7132469.html
-Advertisement-
Play Games

一 概述 1.什麼是動態查詢? 從多個查詢條件中隨機選擇若幹個組合成一個DQL語句進行查詢,這一過程叫做動態查詢。 2.動態查詢的難點 可供選擇的查詢條件多,組合情況多,難以一一列舉。 3.最終查詢語句的構成 一旦用戶向查詢條件中輸入數據,該查詢條件就成為最終條件的一部分。 二 基本原理 1.SQL ...


 

一 概述

1.什麼是動態查詢?

從多個查詢條件中隨機選擇若幹個組合成一個DQL語句進行查詢,這一過程叫做動態查詢。

2.動態查詢的難點

可供選擇的查詢條件多,組合情況多,難以一一列舉。

3.最終查詢語句的構成

一旦用戶向查詢條件中輸入數據,該查詢條件就成為最終條件的一部分。

二 基本原理

1.SQL基本框架

無論查詢條件如何,查詢欄位與資料庫是固定不變的,這些固定不變的內容構成SQL語句的基本框架,如

select column... from table。

2.StringBuilder形成DQL

獲取表單輸入,如果請求參數非空,根據該請求參數生成查詢條件,如“name=?”,“age>?”,將查詢條件追加到基本框架中。利用StringBuilder來追加查詢條件,這時出現一個問題,怎麼判斷生成的查詢條件中是否需要添加“and”?
如果該查詢條件是第一個查詢條件,不需要添加"and",否則需要添加“and”。問題變得複雜起來,每一次生成查詢條件時都需要判斷前面是否存在查詢條件。
我們可以考慮在SQL基本框架中添加一個查詢條件,該查詢條件的存在不影響查詢結果,只充當占位角色,避免動態添加查詢條件時判斷是否需要添加“and”。根據這些要求,這一查詢條件必須恆為真,這裡我們取“1=1”,SQL基本框架就變成了

select column...from table where 1=1

每一個動態查詢條件前段都添加“and”。

3.List集合為占位符賦值

有了DQL語句,接著需要考慮怎麼為占位符賦值。可以在生成查詢條件的同時,將占位符對應的參數收集起來,存入一個有序集合中,這裡選擇List集合,這樣占位符就與List集合中的元素形成了順序上的對應關係,第n個占位符對應第n個元素,遍歷集合就可以為占位符賦值了。
為占位符賦值時,不僅僅需要將數據傳遞給占位符,還需要選擇與欄位一致的數據類型,List集合僅僅存儲數據已經不能夠滿足要求了,還需要添加欄位信息,以區分不同的欄位,選擇不同的數據類型。這裡集合中的元素採用“column+data”的形式。

三 Demo

1.資料庫

2.頁面

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<style>
span {
    display: inline-block;
    width: 75px;
    margin-bottom: 15px;
}
</style>
<title>動態查詢</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>部門編號:</span><input type="text" name="depNo">
        </div>
        <div>
            <input type="submit"value="查詢">&nbsp;<input type="reset"value="重置">
        </div>
    </form>
</body>
</html>

3.伺服器端(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");
        // 獲取請求參數
        String name = request.getParameter("name");
        String sex = request.getParameter("sex");
        String age = request.getParameter("age");
        String depNo = request.getParameter("depNo");

        // 關鍵是"where 1=1",不需要再判斷追加的查詢條件前是否需要添加and,統一在前面添加and
        String baseSQL = "select name,sex,age,depNo from tb_employee where 1=1";
        StringBuilder builder = new StringBuilder();// 用於拼接SQL語句
        // 用於在占位符與參數值之間建立映射,占位符與參數值在各自序列中的排序一相同,例如name的占位符在SQL語句中排第一,name的參數值在
        // 集合中排第一。
        List<String> params = new ArrayList<String>();
        builder.append(baseSQL);
        if (isNotEmpty(name)) {
            builder.append(" and name=? ");
            params.add("name," + name);// 集合中不能僅僅存儲具體的數據,還要存儲欄位名,以便後續根據欄位名選擇數據類型
        }
        if (isNotEmpty(sex)) {
            builder.append(" and sex=? ");
            params.add("sex," + sex);//List集合中不僅存儲了表單輸入數據,而且存儲了對應欄位
        }
        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]儲存欄位信息,用於區分欄位;arr[1]存儲數據,用於為占位符賦值
                // 因為為占位符賦值時,需要根據欄位類型選擇數據類型,所以在此判斷類型
                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("查詢為空");
        else
            out.write(builder.toString() + "<br>" + resStr.toString());
    }

    /**
     * 判斷請求參數是否存在,是否有數據輸入
     * 
     * @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");
    }

}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 實驗環境:OS X EI Captian + MySQL 5.7一、配置MySQL自動記錄慢查詢日誌查看變數,也就是配置信息show (global) variables like '%slow_query%'修改變數:set global 變數名 = 值與慢查詢日誌有關的變數:是否開啟慢查詢日誌s... ...
  • 以下內容適用於oracle 10.2.0.5及其以上版本 一個查詢很慢,原始SQL如下: 表都不大,執行計劃如下: 這個執行計劃,看起來無比正常,應該要left join的都有。 但問題的關鍵在於10 步驟-- MERGE JOIN CARTESIAN。笛卡爾乘積的排序合併連接,這個需要耗費很長時間 ...
  • 首先介紹用到的兩個函數 charindex(要查找的表達式1,表達式2),返回值為表達式1在表達式2中的下標,未找到則返回0。(sql的下標是從1開始的),例如 select charindex('sql','sql server 2012') 返回1 另一個是stuff(要替換的表達式1, 起始位 ...
  • OpenWAF是第一個全方位開源的Web應用防護系統; ELK 是比較火的開源日誌分析系統; 本節主要介紹,ELK 的 docker 部署及與 OpenWAF 的結合 ...
  • 在linux下向oracle資料庫導入資料庫文件 執行語句:imp C##root/123456 file=/home/oracle/dbf/crm.dmp ignore=y full=y; 報錯:IMP-00003: 遇到 ORACLE 錯誤 6550 ORA-06550: 第 1 行, 第 19 ...
  • redis兩種持久化的方式 RDB持久化可以在指定的時間間隔內生成數據集的時間點快照 AOF持久化記錄伺服器執行的所有寫操作命令,併在伺服器啟動時,通過重新執行這些命令來還原數據集,AOF文件中全部以redis協議的格式來保存,新命令會被追加到文件的末尾,redis還可以在後臺對AOF文件進行重寫, ...
  • MySQL是一個關係型資料庫管理系統,由瑞典MySQLAB公司開發,目前屬於Oracle旗下產品。MySQL是最流行的關係型資料庫管理系統之一,在web應用方面,MySQL是最好的RDBMS(RelationalDatabaseManagementSystem,關係資料庫管理系統)應用軟體。 mys ...
  • OracleDatabase,又名OracleRDBMS,或簡稱Oracle。是甲骨文公司的一款關係資料庫管理系統。它是在資料庫領域一直處於領先地位的產品。可以說Oracle資料庫系統是目前世界上流行的關係資料庫管理系統,系統可移植性好、使用方便、功能強,適用於各類大、中、小、微機環境。它是一種高效 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...