JavaWab(四)
本文摘要:
- 用Element组件编写一个好看的页面
- 能够完成查询所有功能
- 能够完成添加功能
- 能够理解 BaseServlet 思想
- 能够完成批量删除功能
- 能够完成分页查询功能
- 能够完成条件查询功能
- 完善了课程中没讲的修改和删除功能
案例页面
那我们现在就开始从上往下找组件来拼凑这个页面,在此之前,我们先创建一个brand_case
的项目,并在webapp
下新建brand.html,引入Element 的css、js文件 和 Vue.js
1 | HTML |
搜索表单的展示
Element-组件-Form表单-行内表单,这个组件可以完成我们的搜索表单,直接把代码Copy过来进行修改
- 行内表单示例代码
- 修改之后的代码
- 测试
将活动区域
换成当前状态
,并将区域一,区域二
换成启用,禁用
,审批人
换成企业名称
,然后在复制一份,改成品牌名称
data和methods都放到我们创建的vue对象中,同时将绑定的对象也改一下
1 | HTML |
批量删除和修改的按钮
Element-组件-Button按钮-基础用法,这个组件可以给我们提供好看的按钮,直接Copy过来
1 | HTML |
新增对话框展示
删除按钮我们暂时不动,但是当我们点击新增
按钮时,需要弹出一个对话框让我们填写信息
Element-组件-Dialog对话框,这个组件可以帮我们达到目的
- 对话框示例代码
- 典型表单示例代码
- 修改之后的代码
- 测试
需要设置visible属性,它接收Boolean,当为true时显示 Dialog。Dialog 分为两个部分:body和footer,footer需要具名为footer的slot。title属性用于定义标题,它是可选的,默认值为空。最后,本例还展示了before-close的用法。
1 | HTML |
表格展示
Element-组件-Table表格可以完成我们的需求
- 示例代码
- 修改之后的代码
1 | HTML |
分页条展示
Element-组件-Pagination 分页可以帮我们完成需求
- 分页示例代码
- 修改之后
1 | HTML |
完整代码如下
1 | HTML |
环境准备
工程准备
在Java目录下创建com.blog.mapper
,com.blog.pojo
,com.blog.service
,com.blog.util
,com.blog.web
这五个包
在mapper包下新建
BrandMapper
接口1
2
3
4
5
6JAVA
import java.util.List;
public interface BrandMapper {
}在resource目录下新建
com/blog/mapper
路径,新建一个BrandMapper.xml
文件1
2
3
4
5
6
7
8
9
10
11
12XML
<mapper namespace="com.blog.mapper.BrandMapper">
<resultMap id="brandResultMap" type="brand">
<result property="brandName" column="brand_name" />
<result property="companyName" column="company_name" />
</resultMap>
</mapper>在resource目录下新建
mybatis-config.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27XML
<configuration>
<!--起别名-->
<typeAliases>
<package name="com.blog.pojo"/>
</typeAliases>
<environments default="development">
<environment id="development">
<transactionManager type="JDBC"/>
<dataSource type="POOLED">
<property name="driver" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:13306/db1?useSSL=false&useServerPrepStmts=true"/>
<property name="username" value="root"/>
<property name="password" value="PASSWORD"/>
</dataSource>
</environment>
</environments>
<mappers>
<!--扫描mapper-->
<package name="com.blog.mapper"/>
</mappers>
</configuration>在pojo包下新建
Brand
类1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83JAVA
public class Brand {
// id 主键
private Integer id;
// 品牌名称
private String brandName;
// 企业名称
private String companyName;
// 排序字段
private Integer ordered;
// 描述信息
private String description;
// 状态:0:禁用 1:启用
private Integer status;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getBrandName() {
return brandName;
}
public void setBrandName(String brandName) {
this.brandName = brandName;
}
public String getCompanyName() {
return companyName;
}
public void setCompanyName(String companyName) {
this.companyName = companyName;
}
public Integer getOrdered() {
return ordered;
}
public void setOrdered(Integer ordered) {
this.ordered = ordered;
}
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
public Integer getStatus() {
return status;
}
//逻辑视图
public String getStatusStr(){
if (status == null){
return "未知";
}
return status == 0 ? "禁用":"启用";
}
public void setStatus(Integer status) {
this.status = status;
}
public String toString() {
return "Brand{" +
"id=" + id +
", brandName='" + brandName + '\'' +
", companyName='" + companyName + '\'' +
", ordered=" + ordered +
", description='" + description + '\'' +
", status=" + status +
'}';
}
}在util包下导入工具类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20JAVA
public class SqlSessionFactoryUtils {
private static SqlSessionFactory sqlSessionFactory;
static {
//静态代码块会随着类的加载而自动执行,且只执行一次
try {
String resource = "mybatis-config.xml";
InputStream inputStream = Resources.getResourceAsStream(resource);
sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
} catch (IOException e) {
e.printStackTrace();
}
}
public static SqlSessionFactory getSqlSessionFactory(){
return sqlSessionFactory;
}
}在pom.xml中导入坐标
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57XML
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>org.example</groupId>
<artifactId>brand_demo</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>war</packaging>
<properties>
<maven.compiler.source>8</maven.compiler.source>
<maven.compiler.target>8</maven.compiler.target>
</properties>
<dependencies>
<!-- mybatis -->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.5</version>
</dependency>
<!--mysql-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.34</version>
</dependency>
<!--servlet-->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>javax.servlet-api</artifactId>
<version>3.1.0</version>
<scope>provided</scope>
</dependency>
<!--fastjson-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.62</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.2</version>
</plugin>
</plugins>
</build>
</project>
创建表
1 | SQL |
查询所有功能
后端实现
dao方法实现
由于表中有些字段名和实体类中的属性名没有对应,所以需要在
com/itheima/mapper/BrandMapper.xml
映射配置文件中定义结果映射 ,使用resultMap
标签。映射配置文件内容如下:1
2
3
4
5
6
7
8
9
10
11
12XML
<mapper namespace="com.blog.mapper.BrandMapper">
<resultMap id="brandResultMap" type="brand">
<result property="brandName" column="brand_name" />
<result property="companyName" column="company_name" />
</resultMap>
</mapper>在
com.blog.mapper.BrandMapper
接口中定义抽象方法,并使用@Select
注解编写SQL语句1
2
3
4
5
6
7
8JAVA
/**
* 查询所有
* @return
*/
List<Brand> selectAll();
service方法实现
在
com.itheima.service
包下创建BrandService
接口,在该接口中定义查询所有的抽象方法1
2
3
4
5
6
7
8JAVA
public interface BrandService {
/**
* 查询所有
* @return
*/
List<Brand> selectAll();
}并在
com.itheima.service
下再创建impl
包;impl
表示是放 service 层接口的实现类的包。 在该包下创建名为BrandServiceImpl
类1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18JAVA
public class BrandServiceImpl implements BrandService {
//1. 创建SqlSessionFactory工厂对象
SqlSessionFactory sqlSessionFactory = SqlSessionFactoryUtils.getSqlSessionFactory();
public List<Brand> selectAll() {
//2. 获取SqlSession对象
SqlSession sqlSession = sqlSessionFactory.openSession();
//3. 获取BrandMapper
BrandMapper mapper = sqlSession.getMapper(BrandMapper.class);
//4. 调用方法
List<Brand> brands = mapper.selectAll();
//5. 关闭资源
sqlSession.close();
return brands;
}
}为什么要创建BrandServiceImpl?
- 因为service定义了接口后,在 servlet 中就可以使用多态的形式创建Service实现类的对象
这里使用多态是因为方便解除Servlet
和service
的耦合。从上面的代码我们可以看到SelectAllServlet
类和BrandServiceImpl
类之间是耦合在一起的,如果后期BrandService
有其它的实现类(例如叫BrandServiceImpl1
),那就需要修改SelectAllServlet
类中的代码。后面学习了Spring
框架后就可以解除SelectAllServlet
类和BrandServiceImpl
的代码耦合。但现在还做不到解耦合,在这里只需要理解为什么定义接口即可。
- 因为service定义了接口后,在 servlet 中就可以使用多态的形式创建Service实现类的对象
servlet方法实现
- 在
1 | com.itheima.web.servlet |
包下定义名为
1 | SelectAllServlet |
的查询所有的
1 | servlet |
。该
1 | servlet |
逻辑如下:
- 调用service的
selectAll()
方法查询所有的品牌数据,并接口返回结果 - 将返回的结果转换为 json 数据
- 响应 json 数据
1 | JAVA |
测试后端程序
在浏览器输入访问 servlet 的资源路径 http://localhost:8080/brand_case/selectAllServlet
,如果没有报错,并能看到如下信息表明后端程序没有问题
[{“brandName”:”华为”,”companyName”:”华为技术有限公司”,”description”:”万物互联”,”id”:1,”ordered”:100,”status”:1,”statusStr”:”启用”},
前端实现
前端需要在页面加载完毕后发送 ajax 请求,所以发送请求的逻辑应该放在 mounted()
钩子函数中。而响应回来的数据需要赋值给表格绑定的数据模型
1 | JS |
添加功能
上图是添加数据的对话框,当点击 提交
按钮后就需要将数据提交到后端,并将数据保存到数据库中。
页面发送请求时,需要将输入框输入的内容提交给后端程序,而这里是以 json 格式进行传递的。
后端实现
dao方法实现
- 在
1 | BrandMapper |
接口中定义
1 | add() |
添加方法,并使用
1 | @Insert |
注解编写sql语句
1 | JAVA |
service方法实现
在
BrandService
接口中定义add()
添加数据的业务逻辑方法1
2
3
4
5
6JAVA
/**
* 添加数据
* @param brand
*/
void add(Brand brand);在
BrandServiceImpl
类中重写add()
方法,并进行业务逻辑实现1
2
3
4
5
6
7
8
9
10
11
12
13JAVA
public void add(Brand brand) {
//2. 获取SqlSession对象
SqlSession sqlSession = sqlSessionFactory.openSession();
//3. 获取BrandMapper
BrandMapper mapper = sqlSession.getMapper(BrandMapper.class);
//4. 调用方法
mapper.add(brand);
sqlSession.commit();//提交事务
//5. 释放资源
sqlSession.close();
}注意:增删改操作一定要提交事务。
servlet方法实现
在
com.blog.web.servlet
包写定义名为AddServlet
的 Servlet。该 Servlet 的逻辑如下:- 接收页面提交的数据。页面到时候提交的数据是 json 格式的数据,所以此处需要使用输入流读取数据
- 将接收到的数据转换为
Brand
对象 - 调用 service 的
add()
方法进行添加的业务逻辑处理 - 给浏览器响应添加成功的标识,这里直接给浏览器响应
success
字符串表示成功
servlet 代码实现如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26JAVA
@WebServlet("/addServlet")
public class AddServlet extends HttpServlet {
private BrandService brandService = new BrandServiceImpl();
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//解决乱码问题,一定要把它放在第一行
request.setCharacterEncoding("utf-8");
//获取输入流
BufferedReader br = request.getReader();
//读取数据
String params = br.readLine();
//将json格式的数据转换为一个Brand对象
Brand brand = JSON.parseObject(params, Brand.class);
//调用方法
brandService.add(brand);
//如果能顺利执行到此行,则说明添加成功,给浏览器响应一个success字符串表示成功
response.getWriter().write("success");
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doGet(request, response);
}
}
前端实现
我们之前给提交按钮的点击事件绑定了一个addBrand
函数,所以添加数据功能的逻辑代码应该写在 addBrand()
函数中。在此方法中需要发送异步请求并将表单中输入的数据作为参数进行传递。如下
1 | JS |
在 then
函数中的匿名函数是成功后的回调函数,而 resp.data
就可以获取到响应回来的数据,如果值是 success
表示数据添加成功。那么添加成功之后有哪些操作呢?
先将对话框关闭,设置dialogVisible为false,就可以关闭对话框了
1
2JS
_this.dialogVisible = false重新查询数据,这部分代码其实就是刚刚查询所有的前端代码,将其封装成一个selectAll()函数
1
2
3
4
5
6
7
8JS
var _this = this;
axios({
method:"get",
url:"http://localhost:8080/brand_case/selectAllServlet"
}).then(function (resp) {
_this.tableData = resp.data;
})给出提示信息,这部分其实是直接从ElementUI官网扒的代码,想用花里胡哨的可以
戳我
1
2
3
4
5JS
_this.$message({
message: '恭喜你,添加成功',
type: 'success'
});
servlet优化
问题导入
Web 层的 Servlet 个数太多了,不利于管理和编写。
每一个功能都需要定义一个 servlet
,一个模块需要实现增删改查功能,就需要4个 servlet
,模块一多就会造成servlet
泛滥。此时我们就想 servlet
能不能像 service
一样,一个模块只定义一个 servlet
,而每一个功能只需要在该 servlet
中定义对应的方法。例如下面代码:
1 | JAVA |
而我们知道发送请求 servlet
,tomcat
会自动的调用 service()
方法,HttpServlet的service()方法如下,当我们访问该 servlet
时会根据请求方式将请求分发给 doGet()
或者 doPost()
等方法
1 | JAVA |
那么我们也可以仿照这样请求分发的思想,在 service()
方法中根据具体的操作调用对应的方法,如:查询所有就调用 selectAll()
方法,添加企业信息就调用 add()
方法。
为了做到通用,我们定义一个通用的 servlet
类,再定义其他的 servlet
就不用继承 HttpServlet
,而是继承我们自定义的 BaseServlet
,在BaseServlet
中调用具体 servlet
(如BrandServlet
)中的对应方法。
1 | JAVA |
BrandServlet定义修改如下
1 | JAVA |
那么如何在 BaseServlet
中调用对应的方法呢?比如查询所有就调用 selectAll()
方法。
可以规定在发送请求时,请求资源的二级路径(/brandServlet/selectAll)和需要调用的方法名相同,如:
查询所有数据的路径以后就需要写成: http://localhost:8080/brand_case/brand/selectAll
添加数据的路径以后就需要写成: http://localhost:8080/brand_case/brand/add
修改数据的路径以后就需要写成: http://localhost:8080/brand_case/brand/update
删除数据的路径以后就需要写成: http://localhost:8080/brand_case/brand/delete
这样的话,在 BaseServlet
中就需要获取到资源的二级路径作为方法名,然后调用该方法
1 | JAVA |
通过上面代码发现根据方法名获取对应方法的 Method
对象时需要指定方法参数的字节码对象。解决这个问题,可以将方法的参数类型规定死,而方法中可能需要用到 request
对象和 response
对象,所以指定方法的参数为 HttpServletRequest
和 HttpServletResponse
,那么 BrandServlet
代码就可以改进为:
1 | JAVA |
BaseServlet代码可以改进为:
1 | JAVA |
代码优化
后端优化
定义了 BaseServlet
后,针对品牌模块我们定义一个 BrandServlet
的 Servlet,并使其继承 BaseServlet
。在BrandServlet
中定义 以下功能的方法:
查询所有
功能:方法名声明为selectAll
,并将之前的SelectAllServlet
中的逻辑代码拷贝到该方法中添加数据
功能:方法名声明为add
,并将之前的AddServlet
中的逻辑代码拷贝到该方法中
具体代码如下:
1 | JAVA |
前端优化
页面中之前发送的请求的路径都需要进行修改,selectAll()
和addBrand()
函数中发送异步请求的 url
应该改为http://localhost:8080/brand_case/brand/selectAll
和http://localhost:8080/brand_case/brand/add
1 | JS |
批量删除
点击多条数据前的复选框就意味着要删除这些数据,而点击了 批量删除
按钮后,需要让用户确认一下,因为有可能是用户误操作的,当用户确定后需要给后端发送请求并携带者需要删除数据的多个id值,前端发送请求时需要将要删除的多个id值以json格式提交给后端
后端实现
dao方法实现
接口方法声明如下
1
2
3
4
5
6
7JAVA
/**
* 批量删除
*
* @param ids
*/
void deleteByIdIs(@Param("ids") int[] ids);在
1 | BrandMapper.xml |
映射配置文件中添加 statement,需要用到动态sql的知识,在前面的
文章
已经详细说过了
1 | XML |
service方法实现
在
BrandService
接口中定义deleteByIds()
批量删除的业务逻辑方法1
2
3
4
5
6JAVA
/**
* 批量删除
* @param ids
*/
void deleteByIds(int[] ids);在
BrandServiceImpl
类中重写deleteByIds()
方法,并进行业务逻辑实现1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16JAVA
@Override
public void deleteByIds(int[] ids) {
//2. 获取SqlSession对象
SqlSession sqlSession = sqlSessionFactory.openSession();
//3. 获取BrandMapper
BrandMapper mapper = sqlSession.getMapper(BrandMapper.class);
//4. 调用方法
mapper.deleteByIds(ids);
sqlSession.commit();//提交事务
//5. 释放资源
sqlSession.close();
}
servlet方法实现
在
BrandServlet
类中定义deleteByIds()
方法。而该方法的逻辑如下:- 接收页面提交的数据。页面到时候提交的数据是 json 格式的数据,所以此处需要使用输入流读取数据
- 将接收到的数据转换为
int[]
数组 - 调用 service 的
deleteByIds()
方法进行批量删除的业务逻辑处理 - 给浏览器响应添加成功的标识,这里直接给浏览器响应
success
字符串表示成功
servlet 中
deleteByIds()
方法代码实现如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25JAVA
/**
* 批量删除
*
* @param request
* @param response
* @throws ServletException
* @throws IOException
*/
public void deleteByIds(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//处理乱码问题
request.setCharacterEncoding("utf-8");
//1. 获取输入流,接收品牌数据json字符串
BufferedReader br = request.getReader();
String params = br.readLine();
//将json字符串转为int[]数组
int[] ids = JSON.parseObject(params, int[].class);
//2. 调用service添加
brandService.deleteByIds(ids);
//3. 响应成功的标识
response.getWriter().write("success");
}
前端实现
先去ElementUI官网看看复选框都有什么属性,发现表单绑定了一个事件
1
@selection-change="handleSelectionChange
1
2
3
4JS
handleSelectionChange(val) {
this.multipleSelection = val;
}该事件是当选择项发生变化时会触发。该事件绑定了
handleSelectionChange
函数,而该函数有一个参数val
,该参数是获取选中行的数据,选中多行也就是相当于选中了一个brand数组,我们可以通过遍历brand数组来获取其id,然后使用 axios 发送异步请求并经刚刚获取到的存储所有的 id 数组作为请求参数,那么现在就分析完毕了,开始敲代码给批量删除绑定一个点击事件,并绑定触发时的调用函数
deleteByIds
1
2HTML
<el-button @click="deleteByIds" type="danger" plain>批量删除</el-button>我们要在
deleteByIds
函数中遍历选中的brand数组,也就是遍历this.multipleSelection
,获取brand.id
,将其加入到id数组,所以我们先在Vue对象的data中添加selectedIds: []
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20JS
//获取选中的id值,并加入到selectedIds数组中
for (let i = 0; i < this.multipleSelection.length; i++) {
this.selectedIds[i] = this.multipleSelection[i].id;
}
var _this = this;
axios({
//发送异步请求提交json数据,用post
method: "post",
url: "http://localhost:8080/test_brand/brand/deleteByIds", //地址就是BrandServlet中的批量删除方法
data: _this.selectedIds //提交的数据就是的selectedIds
}).then(function (resp) {
if (resp.data == "success") { //获取响应标识
_this.selectAll(); //如果成功添加了,就重新查询数据,删除完毕,不用手动刷新页面就能看到删除结果
_this.$message({ //给出提示信息
message: '恭喜你,删除成功',
type: 'success'
});
}
})由于删除操作是一个危险的操作,所以在删除之前,我们需要让用户确认一下是否删除,要完成这个需求,我们继续去ElementUI中找个组件,MessageBox弹框
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31HTML
<template>
<el-button type="text" @click="open">点击打开 Message Box</el-button>
</template>
<script>
export default {
methods: {
open() {
//这里的文案也可以改一下
this.$confirm('此操作将永久删除该文件, 是否继续?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
//当点击确定,就执行这里,将我们刚刚写的删除操作放到这里就可以了
this.$message({
type: 'success',
message: '删除成功!'
});
}).catch(() => {
//点击取消就不用管了
this.$message({
type: 'info',
message: '已取消删除'
});
});
}
}
}
</script>直接把这个open函数的函数体搬走就可以,修改完成的
1
deleteByIds
函数代码如下
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31JS
deleteByIds() {
this.$confirm('此操作将删除数据, 是否继续?', '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
for (let i = 0; i < this.multipleSelection.length; i++) {
this.selectedIds[i] = this.multipleSelection[i].id;
}
var _this = this;
axios({
method: "post",
url: "http://localhost:8080/test_brand/brand/deleteByIds",
data: _this.selectedIds
}).then(function (resp) {
if (resp.data == "success") {
_this.selectAll();
_this.$message({
message: '恭喜你,删除成功',
type: 'success'
});
}
})
}).catch(() => {
this.$message({
type: 'info',
message: '已取消删除'
});
});
}
分页查询
我们之前做的 查询所有
功能中将数据库中所有的数据查询出来并展示到页面上,试想如果数据库中的数据有很多(假设有十几万条)的时候,将数据全部展示出来肯定不现实,那如何解决这个问题呢?几乎所有的网站都会使用分页解决这个问题。每次只展示一页的数据,比如一页展示10条数据,如果还想看其他的数据,可以通过点击页码进行查询
分析
分页查询sql
分页查询也是从数据库进行查询的,所以我们要分页对应的SQL语句应该怎么写。分页查询使用
LIMIT
关键字,格式为:LIMIT 开始索引 每页显示的条数
。以后前端页面在发送请求携带参数时,它并不明确开始索引是什么,但是它知道查询第几页。所以开始索引
需要在后端进行计算,计算的公式是 :开始索引 = (当前页码 - 1)* 每页显示条数比如查询第一页的数据的 SQL 语句是:
1
2SQL
select * from tb_brand limit 0,5;查询第二页的数据的 SQL 语句是:
1
2SQL
select * from tb_brand limit 5,5;查询第三页的数据的 SQL 语句是:
1
2SQL
select * from tb_brand limit 10,5;
前后端数据分析
分页查询功能时候比较复杂的,所以我们要先分析清楚以下两个问题:
前端需要传递什么参数给后端
根据上一步对分页查询 SQL 语句分析得出,前端需要给后端两个参数- 当前页码 :currentPage
- 每页显示条数:pageSize
后端需要响应什么数据给前端
- 当前页需要展示的数据。我们在后端一般会存储到 List 集合中
- 总共记录数。在上图页面中需要展示总的记录数,所以这部分数据也需要。总的页面 ElementUI 的分页组件会自动计算,我们不需要关心
- 将以上两部分封装到 PageBean 对象中,并将该对象转换为 json 格式的数据响应回给浏览器
通过上面的分析我们需要先在
pojo
包下创建PageBean
类,为了做到通用性会将其定义成泛型类,代码如下:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25JAVA
//分页查询的JavaBean
public class PageBean<T> {
// 总记录数
private int totalCount;
// 当前页数据
private List<T> rows;
public int getTotalCount() {
return totalCount;
}
public void setTotalCount(int totalCount) {
this.totalCount = totalCount;
}
public List<T> getRows() {
return rows;
}
public void setRows(List<T> rows) {
this.rows = rows;
}
}
后端实现
dao方法实现
- 在
1 | BrandMapper |
接口中定义
1 | selectByPage() |
方法进行分页查询,代码如下:
1 | JAVA |
- 在
1 | BrandMapper |
接口中定义
1 | selectTotalCount() |
方法进行统计记录数,代码如下:
1 | JAVA |
service方法实现
在 BrandService
接口中定义 selectByPage()
分页查询数据的业务逻辑方法
1 | JAVA |
在 BrandServiceImpl
类中重写 selectByPage()
方法,并进行业务逻辑实现
1 | JAVA |
servlet方法实现
- 在
1 | BrandServlet |
类中定义
1 | selectByPage() |
方法。而该方法的逻辑如下:
- 获取页面提交的
当前页码
和每页显示条目数
两个数据。这两个参数是在url后进行拼接的,格式是url?currentPage=1&pageSize=5
。获取这 样的参数需要使用requet.getparameter()
方法获取。 - 调用 service 的
selectByPage()
方法进行分页查询的业务逻辑处理 - 将查询到的数据转换为 json 格式的数据
- 响应 json 数据
- servlet 中
1 | selectByPage() |
方法代码实现如下:
1 | JAVA |
测试
在浏览器上地址栏输入 http://localhost:8080/brand_case/brand/selectByPage?currentPage=1&pageSize=5
,可以查询到数据
前端实现
selectAll 代码改进
selectAll()
函数之前是查询所有数据,现需要改成分页查询。 请求路径应改为 http://localhost:8080/brand_case/brand/selectByPage?currentPage=1&pageSize=5
,而 currentPage
和 pageSize
是需要携带的参数,分别是 当前页码
和 每页显示的条目数
。
刚才我们对后端代码进行测试可以看出响应回来的数据,所以在异步请求的成功回调函数(then
中的匿名函数)中给页面表格的数据模型赋值。整体代码如下
1 | JS |
改变当前页的条目数和当前页码
先来看看分页条的代码@size-change
就是每页显示的条目数发生变化时会触发的事件,而该事件绑定了一个 handleSizeChange
函数@current-change
就是页码发生变化时会触发的事件,而该事件绑定了一个 handleSizeChange
函数
1 | HTML |
我们只需要在这两个函数中重新设置当前页的条目数和当前页码,然后调用selectAll
函数重新分页查询数据
1 | JS |
条件查询
要做条件查询功能,先明确以下三个问题
- 3个条件之间什么关系?
- 同时满足,所用 SQL 中多个条件需要使用 and 关键字连接
- 3个条件必须全部填写吗?
- 不需要。想根据哪儿个条件查询就写那个,所以这里需要使用动态 sql 语句
- 条件查询需要分页吗?
- 需要
后端实现
dao实现
在 BrandMapper
接口中定义 selectByPageAndCondition()
方法 和 selectTotalCountByCondition
方法,用来进行条件分页查询功能,方法如下:
1 | JAVA |
参数:
begin
分页查询的起始索引size
分页查询的每页条目数brand
用来封装条件的对象
由于这是一个复杂的查询语句,需要使用动态sql;所以我们在映射配置文件中书写 sql 语句。brand_name
字段和 company_name
字段需要进行模糊查询,所以需要使用 %
占位符。映射配置文件中 statement 书写如下:
1 | XML |
service实现
在 BrandService
接口中定义 selectByPageAndCondition()
分页查询数据的业务逻辑方法
1 | JAVA |
在 BrandServiceImpl
类中重写 selectByPageAndCondition()
方法,并进行业务逻辑实现
1 | JAVA |
注意:brandName 和 companyName 属性值到时候需要进行模糊查询,所以前后需要拼接上 %
servlet实现
在 BrandServlet
类中定义 selectByPageAndCondition()
方法。而该方法的逻辑如下:
获取页面提交的
当前页码
和每页显示条目数
两个数据。这两个参数是在url后进行拼接的,格式是url?currentPage=1&pageSize=5
。获取这样的参数需要使用requet.getparameter()
方法获取。获取页面提交的
条件数据
,并将数据封装到一个Brand对象中。由于这部分数据到时候是需要以 json 格式进行提交的,所以我们需要通过流获取数据,具体代码如下:1
2
3
4
5
6
7JAVA
// 获取查询条件对象
BufferedReader br = request.getReader();
String params = br.readLine();//json字符串
//转为 Brand
Brand brand = JSON.parseObject(params, Brand.class);调用 service 的
selectByPageAndCondition()
方法进行分页查询的业务逻辑处理将查询到的数据转换为 json 格式的数据
响应 json 数据
servlet 中 selectByPageAndCondition()
方法代码实现如下:
1 | JAVA |
前端实现
前端代码我们从以下几方面实现:
查询表单绑定查询条件对象模型
这一步在页面上已经实现了,页面代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19HTML
<el-form :inline="true" :model="brand" class="demo-form-inline">
<el-form-item label="当前状态">
<el-select v-model="brand.status" placeholder="当前状态">
<el-option label="启用" value="1"></el-option>
<el-option label="禁用" value="0"></el-option>
</el-select>
</el-form-item>
<el-form-item label="企业名称">
<el-input v-model="brand.companyName" placeholder="企业名称"></el-input>
</el-form-item>
<el-form-item label="品牌名称">
<el-input v-model="brand.brandName" placeholder="品牌名称"></el-input>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="onSubmit">查询</el-button>
</el-form-item>
</el-form>点击查询按钮查询数据
从上面的代码可以看到给查询
按钮绑定了onSubmit()
函数,而在onSubmit()
函数中只需要调用selectAll()
函数进行条件分页查询。改进 selectAll() 函数
子页面加载完成后发送异步请求,需要携带当前页码、每页显示条数、查询条件对象。接下来先对携带的数据进行说明:当前页码
和每页显示条数
这两个参数我们会拼接到 URL 的后面查询条件对象
这个参数需要以 json 格式提交给后端程序修改
1
selectAll()
函数逻辑为
1
2
3
4
5
6
7
8
9
10
11
12
13
14
JS
var _this = this;
axios({
method:"post",
url:"http://localhost:8080/brand_case/brand/selectByPageAndCondition?currentPage="+this.currentPage+"&pageSize="+this.pageSize,
data:this.brand
}).then(function (resp) {
//设置表格数据
_this.tableData = resp.data.rows; // {rows:[],totalCount:100}
//设置总记录数
_this.totalCount = resp.data.totalCount;
})
前端代码优化
咱们已经将所有的功能实现完毕。而针对前端代码中的发送异步请求的代码,如下
1 | JS |
需要在成功的回调函数(也就是then
函数中的匿名函数)中使用this,都需要在外边使用 _this
记录一下 this
所指向的对象;因为在外边的 this
表示的是 Vue 对象,而回调函数中的 this
表示的不是 vue 对象。这里我们可以使用 ECMAScript6
中的新语法(箭头函数)来简化这部分代码,如上面的代码可以简化为:
1 | JS |
箭头函数语法:
1 | PLAINTEXT |
箭头函数的作用:
替换(简化)匿名函数。
补充
由于课程中没有删除和修改的功能,所以我这里完善了一下
删除功能
删除功能的难点在于如果获取本行数据的id,但成功获取id之后的操作就与批量删除的操作一样了
批量删除是选中多行数据,然后将多行数据的id放到ids数组中,那么单行数据就只需要将当前行的id放到ids数组中
具体操作就是this.selectedIds[0] = this.currentRow.id;
,前端代码也只需要将for循环替换掉
那现在我们只需要复制一份deleteByIds
,然后改个名,将for循环替换掉,就大功告成了
1 | JS |
如何获取当前行数据
其实我们在写前端页面的时候,就已经遇到过了,那现在我们仔细看看
Table 组件提供了单选的支持,只需要配置highlight-current-row属性即可实现单选。之后由current-change事件来管理选中时触发的事件,它会传入currentRow,oldCurrentRow。如果需要显示索引,可以增加一列el-table-column,设置type属性为index即可显示从 1 开始的索引号。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69 HTML
<template>
<el-table
ref="singleTable"
:data="tableData"
highlight-current-row
@current-change="handleCurrentChange"
style="width: 100%">
<el-table-column
type="index"
width="50">
</el-table-column>
<el-table-column
property="date"
label="日期"
width="120">
</el-table-column>
<el-table-column
property="name"
label="姓名"
width="120">
</el-table-column>
<el-table-column
property="address"
label="地址">
</el-table-column>
</el-table>
<div style="margin-top: 20px">
<el-button @click="setCurrent(tableData[1])">选中第二行</el-button>
<el-button @click="setCurrent()">取消选择</el-button>
</div>
</template>
<script>
export default {
data() {
return {
tableData: [{
date: '2016-05-02',
name: '王小虎',
address: '上海市普陀区金沙江路 1518 弄'
}, {
date: '2016-05-04',
name: '王小虎',
address: '上海市普陀区金沙江路 1517 弄'
}, {
date: '2016-05-01',
name: '王小虎',
address: '上海市普陀区金沙江路 1519 弄'
}, {
date: '2016-05-03',
name: '王小虎',
address: '上海市普陀区金沙江路 1516 弄'
}],
currentRow: null
}
},
methods: {
setCurrent(row) {
this.$refs.singleTable.setCurrentRow(row);
},
//这个方法直接搬走
handleCurrentChange(val) {
this.currentRow = val;
}
}
}
</script>
提取一下信息就是我们要在table中配置highlight-current-row
属性,current-change
事件,和事件绑定的handleCurrentChange
函数,然后在Vue的data中配置currentRow
属性(默认为null),然后就大功告成了
table标签的修改如下
1
2
3
4
5
6
7
8
9HTML
<el-table
:data="tableData"
style="width: 100%"
:row-class-name="tableRowClassName"
highlight-current-row
@current-change="handleCurrentChangeRow"
@selection-change="handleSelectionChange">
<!--我这里给绑定的事件改了个名-->data中新增属性(示例代码中已经提供好了)
1
2JS
currentRow: nullmethods中新增方法(示例代码中已经提供好了,直接CV,我这里改了个名)
1
2
3
4JS
handleCurrentChangeRow(val) {
this.currentRow = val;
}
修改功能
- 修改功能依旧是分为
回显
和修改
两部分 - 修改功能的需求就是当我们点击
修改
按钮(给修改按钮绑定echo()
回显函数)时,会弹出一个对话框,里面是已经填好了的品牌信息,我们只需要修改我们需要改的部分,其他部分不用动,然后点击提交按钮
(给提交按钮绑定update()
更新函数),完成修改功能
前端页面
前端页面我们直接照抄新增页面即可,注意将设置
1
:visible.sync="updateVisible"
,同时在Vue对象的data中,新增updateVisible属性
1
updateVisible: false
,该属性默认为false,用来控制修改对话框是否显示
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29HTML
<!--对话框弹出修改品牌-->
<el-dialog
title="修改品牌"
:visible.sync="updateVisible"
width="30%">
<el-form ref="form" :model="brand" label-width="80px">
</el-form-item>
<el-form-item label="企业名称">
<el-input v-model="brand.companyName"></el-input>
</el-form-item>
<el-form-item label="品牌名称">
<el-input v-model="brand.brandName"></el-input>
</el-form-item>
<el-form-item label="排序">
<el-input v-model="brand.ordered"></el-input>
</el-form-item>
<el-form-item label="状态">
<el-switch v-model="brand.status" active-value=1 inactive-value=0></el-switch>
</el-form-item>
<el-form-item label="备注">
<el-input type="textarea" v-model="brand.description"></el-input>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="update">提交</el-button>
<el-button @click="updateVisible = false">取消</el-button>
</el-form-item>
</el-form>
</el-dialog>
回显功能
回显功能是要当我们点击
修改
按钮时,弹出对话框,同时显示原有数据,那我们就给修改按钮的点击事件绑定一个回显函数而回显函数的功能就是刚刚说的,弹出对话框(设置
updateVisible为true
),显示原有数据(将本行数据赋给brand即可,因为修改对话框已经绑定了brand模型)代码如下
1
2
3
4
5JS
echo(){
this.updateVisible = true;
this.brand = this.currentRow;
}
修改功能
dao实现
接口方法声明如下
1
2
3
4
5
6
7
8JAVA
/**
* 更新数据
* @param brand
*/
@Update("update tb_brand set brand_name=#{brandName},company_name=#{companyName},ordered=#{ordered},`description`=#{description},`status`=#{status} where id =#{id}")
@ResultMap("brandResultMap")
void update(Brand brand);service实现
在
1 | BrandService |
接口中定义
1 | update() |
更新数据的业务逻辑方法
1 | JAVA |
- 在
1 | BrandServiceImpl |
类中重写
1 | update() |
方法,并进行业务逻辑实现
1 | JAVA |
servlet实现
- servlet 中
1 | update() |
方法代码实现如下:
1 | JAVA |
前端实现
前面当我们点击
修改
按钮时,已经帮我们弹出对话框,且对话框中有原有的数据,当我们修改完毕之后,点击提交
按钮,就会帮我们完成更新操作
那现在我们只需要来写一个update()
函数就完事儿了,而update()函数和addBrand()函数几乎没有区别,只需要将url改成update,然后提示信息改一下,就大功告成了(区别就是底层的SQL语句不一样,addBrand()执行insert,update()执行update)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18JS
update() {
var _this = this;
axios({
method: "post",
url: "http://localhost:8080/test_brand/brand/update",
data: _this.brand
}).then(function (resp) {
if (resp.data == "success") {
_this.updateVisible = false
_this.selectAll();
_this.$message({
message: '恭喜你,修改成功',
type: 'success'
});
}
})
}至此我们就完成了课程中没讲的修改和删除功能啦
完结撒花
JavaWeb的学习到此就结束了,后面就是学一些主流的框架了,最近先好好复盘一遍JavaWeb巩固基础再往后学。