Java中JDBC实现动态查询的实例详解
一概述
1.什么是动态查询?
从多个查询条件中随机选择若干个组合成一个DQL语句进行查询,这一过程叫做动态查询。
2.动态查询的难点
可供选择的查询条件多,组合情况多,难以一一列举。
3.最终查询语句的构成
一旦用户向查询条件中输入数据,该查询条件就成为最终条件的一部分。
二基本原理
1.SQL基本框架
无论查询条件如何,查询字段与数据库是固定不变的,这些固定不变的内容构成SQL语句的基本框架,如
selectcolumn...fromtable。
2.StringBuilder形成DQL
获取表单输入,如果请求参数非空,根据该请求参数生成查询条件,如“name=?”,“age>?”,将查询条件追加到基本框架中。利用StringBuilder来追加查询条件,这时出现一个问题,怎么判断生成的查询条件中是否需要添加“and”?
如果该查询条件是第一个查询条件,不需要添加"and",否则需要添加“and”。问题变得复杂起来,每一次生成查询条件时都需要判断前面是否存在查询条件。
我们可以考虑在SQL基本框架中添加一个查询条件,该查询条件的存在不影响查询结果,只充当占位角色,避免动态添加查询条件时判断是否需要添加“and”。根据这些要求,这一查询条件必须恒为真,这里我们取“1=1”,SQL基本框架就变成了
selectcolumn...fromtablewhere1=1
每一个动态查询条件前段都添加“and”。
3.List集合为占位符赋值
有了DQL语句,接着需要考虑怎么为占位符赋值。可以在生成查询条件的同时,将占位符对应的参数收集起来,存入一个有序集合中,这里选择List集合,这样占位符就与List集合中的元素形成了顺序上的对应关系,第n个占位符对应第n个元素,遍历集合就可以为占位符赋值了。
为占位符赋值时,不仅仅需要将数据传递给占位符,还需要选择与字段一致的数据类型,List集合仅仅存储数据已经不能够满足要求了,还需要添加字段信息,以区分不同的字段,选择不同的数据类型。这里集合中的元素采用“column+data”的形式。
三Demo
1.数据库
2.页面
动态查询 姓名: 性别: 年龄: 部门编号:
3.服务器端(Servlet)
packagecom.javase.jdbc;
importjava.io.IOException;
importjava.io.PrintWriter;
importjava.sql.Connection;
importjava.sql.DriverManager;
importjava.sql.PreparedStatement;
importjava.sql.ResultSet;
importjava.sql.SQLException;
importjava.util.ArrayList;
importjava.util.List;
importjavax.servlet.ServletException;
importjavax.servlet.annotation.WebServlet;
importjavax.servlet.http.HttpServlet;
importjavax.servlet.http.HttpServletRequest;
importjavax.servlet.http.HttpServletResponse;
@WebServlet("/dynamicQueryServlet")
publicclassDynamicQueryServletextendsHttpServlet{
privatestaticfinallongserialVersionUID=1L;
@Override
protectedvoiddoGet(HttpServletRequestrequest,HttpServletResponseresponse)
throwsServletException,IOException{
response.setContentType("text/html;charset=UTF-8");
//获取请求参数
Stringname=request.getParameter("name");
Stringsex=request.getParameter("sex");
Stringage=request.getParameter("age");
StringdepNo=request.getParameter("depNo");
//关键是"where1=1",不需要再判断追加的查询条件前是否需要添加and,统一在前面添加and
StringbaseSQL="selectname,sex,age,depNofromtb_employeewhere1=1";
StringBuilderbuilder=newStringBuilder();//用于拼接SQL语句
//用于在占位符与参数值之间建立映射,占位符与参数值在各自序列中的排序一相同,例如name的占位符在SQL语句中排第一,name的参数值在
//集合中排第一。
Listparams=newArrayList();
builder.append(baseSQL);
if(isNotEmpty(name)){
builder.append("andname=?");
params.add("name,"+name);//集合中不能仅仅存储具体的数据,还要存储字段名,以便后续根据字段名选择数据类型
}
if(isNotEmpty(sex)){
builder.append("andsex=?");
params.add("sex,"+sex);
}
if(isNotEmpty(age)){
builder.append("andage=?");
params.add("age,"+age);
}
if(isNotEmpty(depNo)){
builder.append("anddepNo=?");
params.add("depNo,"+depNo);
}
Connectionconn=null;
PreparedStatementps=null;
ResultSetres=null;
StringBuilderresStr=newStringBuilder();
try{
conn=getConnection();
ps=conn.prepareStatement(builder.toString());
for(inti=0;i");
}
}catch(ClassNotFoundException|SQLExceptione){
e.printStackTrace();
}finally{
if(res!=null)
try{
res.close();
}catch(SQLExceptione){
e.printStackTrace();
}
if(ps!=null)
try{
ps.close();
}catch(SQLExceptione){
e.printStackTrace();
}
if(conn!=null)
try{
conn.close();
}catch(SQLExceptione){
e.printStackTrace();
}
}
PrintWriterout=response.getWriter();
intlength=resStr.length();
if(length==0)
out.write("查询为空");
else
out.write(builder.toString()+"
"+resStr.toString());
}
/**
*判断请求参数是否存在,是否有数据输入
*
*@paramstr
*@return
*/
privatebooleanisNotEmpty(Stringstr){
if(str==null|str.equals("")){
returnfalse;
}
returntrue;
}
publicstaticConnectiongetConnection()throwsClassNotFoundException,SQLException{
Class.forName("com.mysql.jdbc.Driver");
returnDriverManager.getConnection("jdbc:mysql://localhost:3366/test01","root","123");
}
}
以上所述是小编给大家介绍的Java中JDBC实现动态查询的实例详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!