在网络开发中,多种条件的综合查询非常常见,应对这种业务需求我们通常使用下面几种方法来实现:
1.直接将参数值拼接到sql语句中,然后进行查询。
这种方式的安全性应当说是比较差的,一不小心就被sql注入了。虽然可以先过滤参数值中的特殊字符,但总感觉不是很优雅。
2.先使用占位符'?'来拼接sql,然后再通过条件判断去填充的preparedstatement。
用过这种方式的tx,都知道这种方式的复杂性。先要在拼sql时判断一次,然后还要在填充pst时再判断一次,麻烦。
3.过程存储
的档案本人一直不爽的存储过程,以前有一个项目从mysql的迁移到mssql,后来又换成oracle,最后产品的不同版本运行在不同数据库上,当时差点要了亲命了。
其实讲 么多无非就是想要一种相对优雅简单的查询方式,前一段看到.net中的提供sqlhelper受到一些启发,然后就写了这么一个类似的组件(其实我谷歌了半个小时都没有找到符合要求的基础
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="查询"> <input type="reset"value="重置"></div></form></body></html>

3.服务器端(Servlet)
"/dynamicQueryServlet" DynamicQueryServlet serialVersionUID = 1L "text/html;charset=UTF-8"String name = request.getParameter("name"= request.getParameter("sex"= request.getParameter("age"= request.getParameter("depNo"String baseSQL = "select name,sex,age,depNo from tb_employee where 1=1"= StringBuilder();List<String> params = ArrayList<String>" and name=? ""name," + name);" and sex=? ""sex," +" and age=? ""age," +" and depNo=?""depNo," += = = = == ( i = 0; i < params.size(); i++== str.split(","); (arr[0].equals("age" a = Integer.parseInt(arr[1+ 1+ 1, arr[1== res.getString("name"= res.getString("sex" targetAge = res.getInt("age"= res.getString("depNo"= "name=" + targetName + "--" + "sex=" + targetSex + "--" + "age=" + targetAge + "--"
+ "depNo=" ++ "<br>" (ClassNotFoundException | (res != (ps != (conn != = length = (length == 0"查询为空"+ "<br>" + (str == | str.equals("" Connection getConnection() "com.mysql.jdbc.Driver" DriverManager.getConnection("jdbc:mysql://localhost:3366/test01", "root", "123"
编辑推荐:
- JDBC是如何实现动态查询的?02-28
- Redis的安装与使用02-28
- MySQL中字符串函数的详解02-28
- MySQL自动记录慢查询日志实例02-28
- CentOS服务管理全解析(小白也能轻松掌握chkconfig命令)02-28
- mongodb 3.4下远程连接认证失败应该怎么办?02-28
- MySQL中常用日期时间函数有哪些02-28
- windows下安装mongodb实例教程02-28
相关推荐
-
雷神推出 MIX PRO II 迷你主机:基于 Ultra 200H,玻璃上盖 + ARGB 灯效
2 月 9 日消息,雷神 (THUNDEROBOT) 现已宣布推出基于英
-
制造商 Musnap 推出彩色墨水屏电纸书 Ocean C:支持手写笔、第三方安卓应用
2 月 10 日消息,制造商 Musnap 现已在海外推出一款 Oce
热文推荐
- JDBC是如何实现动态查询的?
JDBC是如何实现动态查询的?
26-02-28 - Redis的安装与使用
Redis的安装与使用
26-02-28 - MySQL中字符串函数的详解
MySQL中字符串函数的详解
26-02-28 - MySQL自动记录慢查询日志实例
MySQL自动记录慢查询日志实例
26-02-28 - CentOS服务管理全解析(小白也能轻松掌握chkconfig命令)
CentOS服务管理全解析(小白也能轻松掌握chkconfig命令)
26-02-28 - mongodb 3.4下远程连接认证失败应该怎么办?
mongodb 3.4下远程连接认证失败应该怎么办?
26-02-28 - MySQL中常用日期时间函数有哪些
MySQL中常用日期时间函数有哪些
26-02-28 - windows下安装mongodb实例教程
windows下安装mongodb实例教程
26-02-28 - 分享关于MySQ函数的实例介绍
分享关于MySQ函数的实例介绍
26-02-28 - 读懂系统负载:轻松掌握CentOS中的/proc/loadavg(小白也能看懂的Linux性能监控指南)
