在设计URL时,使用{}
的占位符时,可以在名称右侧添加:
,并在其右侧配置正则表达式,以对URL中的参数的基本格式进行约束,例如:
// http://localhost:9080/album/9527/delete
@RequestMapping("/{id:[0-9]+}/delete")
public String delete(@PathVariable Long id) {
log.debug("开始处理删除id={}请求", id);
return "处理了/" + id + "/delete的请求";
}
通过使用以上正则表达式,纯数字的id可以匹配以上路径,可以正常访问,如果不是纯数字的id,则根本匹配不到以上路径,以上方法也不会执行,服务器端将直接响应404错误。
提示:404错误相比400错误,能更早的回绝客户端的错误请求。
在使用{}
占位符且使用了正则表达式时,不冲突的匹配(每个URL只会匹配到其中某1个正则表达式,不会同时匹配到多个正则表达式)是可以共存的,例如:
// http://localhost:9080/album/9527/delete
@RequestMapping("/{id:[0-9]+}/delete")
public String delete(@PathVariable Long id) {
log.debug("开始处理删除id={}请求", id);
return "处理了/" + id + "/delete的请求";
}
// http://localhost:9080/album/huawei/delete
@RequestMapping("/{name:[a-zA-Z]+}/delete")
public String delete(@PathVariable String name) {
log.debug("开始处理删除name={}请求", name);
return "处理了/" + name + "/delete的请求";
}
甚至,不使用正则表达式的,也可以与之共存,例如,在以上基础上,还可以添加:
// http://localhost:9080/album/test/delete
@RequestMapping("/test/delete")
public String delete() {
log.debug("开始处理测试删除请求");
return "处理了测试删除的请求";
}
Spring MVC在处理时,会优先匹配没有使用正则表达式的,所以,当提交 /album/test/delete
时,会成功匹配到以上delete()
方法,不会匹配到delete(String name)
方法。
在RESTful的建议中,对于不同的数据操作,应该使用不同的请求类型,例如:
GET >>> /albums/9
:对id值为9的相册数据执行查询(执行数据的select
操作)PUT >>> /albums/9
:对id值为9的相册数据执行编辑(执行数据的update
操作)DELETE >>> /albums/9
:对id值为9的相册数据执行删除(执行数据的delete
操作)POST >>> /albums
:新增相册数据(执行数据的insert
操作)通常,绝大部分应用中,在处理业务时(并不是直接操作某数据),并不会采纳以上建议!
最后,在开发实践中,更多的还是只使用GET
和POST
这2种请求方式,关于RESTful 风格的URL设计参考:
/数据类型的复数
/albums
/数据类型的复数/id值
/albums/{id}
/数据类型的复数/id值/操作
/albums/{id}/delete
MVC = Model + View + Controller
MVC为设计软件提供了基本的思想,它认为每个软件都应该至少包含这3大部分,且各部分分工明确,只负责整个数据处理流程中的一部分功能。
例如V通常表现为“软件的界面”,用于呈现数据、提供用户操作的控件。
而C表示控制器,用于接收请求、响应结果,并不会处理实质业务。
而M表示数据模型,通常由业务逻辑和数据访问这2部分组成,在开发实践中,数据访问通常指的就是数据库编程,而业务逻辑是由专门的类来实现的,这样的类通常使用Service
作为类名的关键字。
在整个数据处理过程中,将会是:Controller调用Service,而Service调用Mapper。
业务逻辑的主要职责是:设计业务流程,处理业务逻辑,以保证数据的完整性和安全性。
Service的开发规范是先写接口,再写实现类。
通常,会在项目的根包下创建service
子包,Service接口将存放在这个包中,并且,还会在service
包下创建impl
子包,Service实现类都将放在这个包中,实现类都会使用ServiceImpl
作为类名的后缀。
例如:在项目的根包下创建service.IAlbumService
接口,然后,再创建service.impl.AlbumServiceImpl
类,且此类将实现IAlbumService
接口。
为了保证项目启动时可以正确的创建此实现类,需要类上添加@Service
注解。
package cn.tedu.csmall.product.service;
public interface IAlbumService {
}
package cn.tedu.csmall.product.service.impl;
import cn.tedu.csmall.product.service.IAlbumService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
@Slf4j
@Service
public class AlbumServiceImpl implements IAlbumService {
public AlbumServiceImpl() {
log.info("创建业务对象:AlbumServiceImpl");
}
}
假设需要实现:添加相册。
则在接口中添加抽象方法,关于抽象方法的设计:
void addNew(AlbumAddNewDTO albumAddNewDTO);
然后,在实现类中:
@Override
public void addNew(AlbumAddNewDTO albumAddNewDTO) {
log.debug("开始处理【添加相册】的业务,参数:{}", albumAddNewDTO);
// 调用AlbumMapper对象的int countByName(String name)方法统计此名称的相册的数量
String name = albumAddNewDTO.getName();
int countByName = albumMapper.countByName(name);
log.debug("尝试添加的相册名称是:{},在数据库中此名称的相册数量为:{}", name, countByName);
// 判断统计结果是否大于0
if (countByName > 0) {
// 是:相册名称已经存在,抛出RuntimeException异常
String message = "添加相册失败!相册名称【" + name + "】已存在!";
log.warn(message);
throw new RuntimeException(message);
}
// 获取当前时间:LocalDateTime now = LocalDateTime.now()
LocalDateTime now = LocalDateTime.now();
// 创建Album对象
Album album = new Album();
// 补全Album对象中各属性的值:name:来自参数
// 补全Album对象中各属性的值:description:来自参数
// 补全Album对象中各属性的值:sort:来自参数
BeanUtils.copyProperties(albumAddNewDTO, album);
// 补全Album对象中各属性的值:gmtCreate:now
album.setGmtCreate(now);
// 补全Album对象中各属性的值:gmtModified:now
album.setGmtModified(now);
// 调用AlbumMapper对象的int insert(Album album)方法插入相册数据
log.debug("即将向数据库中插入数据:{}", album);
albumMapper.insert(album);
}
通常,建议自定义异常,用于表示在业务逻辑层中的“失败”(或错误),而不要使用已知的异常类型,避免捕获、处理不准确!
可以在项目的根包下创建ex.ServiceException
类,继承自RuntimeException
:
public class ServiceException extends RuntimeException {
// 生成5个构造方法
}
Spring MVC框架提供了统一处理异常的机制,使得每种类型的异常在处理时,只需要编写1次相关代码即可。
通常,统一处理异常的代码会写在专门的类中,此类应该添加@ControllerAdvice
,则类中相关的方法会在处理每个请求时生效!
由于目前采取前后端分离的模式,处理异常后的响应方式是响应正文,所以,还应该使用@ResponseBody
,或者,使用@RestControllerAdvice
,它同时具有@ControllerAdvice
和@ResponseBody
的效果。
@RestControllerAdvice
public class GlobalExceptionHandler {
}
然后,在此类中添加处理异常的方法:
@ExceptionHandler
注解,表示此方法是统一处理异常的方法public
HttpServletRequest
、HttpServletResponse
等少量特定类型的参数所以,完整的处理异常的代码为:
package cn.tedu.csmall.product.ex.handler;
import cn.tedu.csmall.product.ex.ServiceException;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.RestControllerAdvice;
@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler
public String handleServiceException(ServiceException e) {
log.debug("处理ServiceException:{}", e.getMessage());
return e.getMessage();
}
}
并且,在任何控制器类中,都不必再处理ServiceException
了。
另外,在以上类中,可以同时存在多个处理不同异常的方法(允许多个处理的异常之间存在继承关系)!
建议在每个项目中,在统一处理异常的类中,都添加对Throwable
的处理,以保证所有异常都会被处理,粗糙的异常信息不会响应到客户端去!
package cn.tedu.csmall.product.ex.handler;
import cn.tedu.csmall.product.ex.ServiceException;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
@Slf4j
@RestControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler
public String handleServiceException(ServiceException e) {
log.debug("处理ServiceException:{}", e.getMessage());
return e.getMessage();
}
@ExceptionHandler
public String handleThrowable(Throwable e) {
log.debug("处理Throwable");
e.printStackTrace();
return "程序运行过程中出现意外错误,请联系系统管理员!";
}
}
Spring Validation框架的主要作用:实现了简化检查请求参数的基本格式
在Spring Boot中,需要添加spring-boot-starter-validation
依赖项。
当需要检查请求参数时,需要在处理请求的方法的参数列表中,对需要检查的参数添加@Validated
注解,表示此参数是需要通过Spring Validation进行检查的:
@RequestMapping("/add-new")
public String addNew(@Validated AlbumAddNewDTO albumAddNewDTO) {
// 省略方法体的代码
}
然后,在类的属性上,添加相关检查注解,并在检查注解中配置message
属性以指定错误时的提示文本:
@Data
public class AlbumAddNewDTO implements Serializable {
@NotNull(message = "必须提交相册名称!")
private String name;
private String description;
private Integer sort;
}
当Spring Validation检查不通过时,将抛出BindException
,所以,可以在统一处理异常的类中对此类异常进行处理:
@ExceptionHandler
public String handleBindException(BindException e) {
log.debug("处理BindException:{}", e.getMessage());
StringBuilder stringBuilder = new StringBuilder();
List fieldErrors = e.getFieldErrors();
for (FieldError fieldError : fieldErrors) {
String message = fieldError.getDefaultMessage();
stringBuilder.append(message);
}
return stringBuilder.toString();
}
除了@NotNull
以外,框架还提供了许多检查注解,
@Pattern
:通过此注解的regexp
属性配置正则表达式,并使用message
配置验证失败时的提示文本
null
”的情况,如果不允许为null
,则必须同时配置@NotNull
和@Pattern
@Range
:通过此注解的min
和max
属性可以指定整型数据的最小值和最大值
@NotNull
一起使用关于根据id删除数据,在处理业务时,应该先根据id查询数据,检查此数据是否存在,然后再删除。
完成各数据的添加和根据id删除,包含Mapper层、业务逻辑层、控制器层。