文章目录
数据库设计
公共字段填充
问题分析
在上一章节已经完成了员工管理和菜品分类功能的开发,在新增员工和新增菜品分类时需要在service的实现类中设置创建时间、创建人、修改时间、修改人等字段,在编辑员工和编辑菜品分类时需要设置修改时间。修改人等字段。这些字段属于公共字段,也就是在我们该项目中很多表中都会有这些字段,如下:
序号 | 字段名 | 含义 | 数据类型 |
---|---|---|---|
1 | create_time | 创建时间 | datetime |
2 | create_user | 创建人id | bigint |
3 | update_time | 修改时间 | datetime |
4 | update_user | 修改人id | bigint |
而针对这些字段,我们的赋值方式为:
- 在新增数据时,将createTime、updateTime设置为当前时间,createUser、updateUser设置为当前登录ID。
- 在更新数据时,将createTime设置为当前时间,createUser设置为当前登录ID。
问题:如果都按照这样的方式来处理这些公共字段,需要在每一个业务方法中进行操作,编码相对冗余、繁琐,那能不能对于这些公共字段在某个地方进行统一处理,来简化开发呢?
答案:我们可以使用AOP编程,实现功能增强,来完成公共字段自动填充功能。
实现思路
- 对于Aop,我们使用过它是对某个类中的方法进行加强,而现在我们是对多个类中的新增和修改的方法来进行加强,那么切入点该怎么写呢,我们可以使用注解
- 我们可以在每个类中新增和修改的方法加上注解,将其设置为方法上注解,但是该怎么区分一个方法是新增还是修改方法呢?我们可以定义约束,来使用枚举
- 我们可以在枚举类中定义俩个值用来区分新增或者修改。(也可以定义默认值,设置一个值,总之能进行区分即可)
- 那么在controller、service、mapper这三层中我们该在哪个类中的新增和修改方法加注解呢?controller的方法中的参数是用来接收前端传过来的属性,而service是controller传过来的,mapper层的参数与数据表中的字段对应,所以我们在mapper层中的新增或修改方法加注解
- 在aop的切面类中我们可以在方法执行前进行填充。
- 在前置通知中JoinPoint可以获取到进行增强方法的参数和方法对象,用获取到的来进行反射给公共字段进行填充
代码演示
- 枚举
- 注解
- 增强类
- mapper方法注解
新增菜品-文件上传
在新增菜品时,需要上传菜品对应的图片(文件),包括后续其他功能也会使用到文件上传,故要实现通用的文件上传接口
介绍
文件上传:是指将本地图片、视频、音频等文件上传到服务器上,可以供其他用户浏览或下载的过程。文件上传在项目中应用非常广泛,我们经常发抖音、发朋友圈都用到了文件上传功能
实现方式
- 直接将图片保存到服务的硬盘(springmvc中的文件上传)
- 优点:开发便捷,成本低
- 缺点:扩容困难
- 使用分布式文件系统进行存储
- 优点:容易实现扩容
- 缺点:开发复杂度稍大(有成熟的产品可以使用:FastDFS、MinIO)
- 使用第三方的存储服务(例如OSS)
- 优点:开发简单,拥有强大功能,免维护
- 缺点:付费
前端使用文件上传三要素
使用阿里云OSS服务进行文件存储
阿里云OSS
阿里云对象存储OSS(Object Storage Service):是一款海量、安全、低成本、高可靠的云存储服务。使用OSS,我们可以通过网络随时存储和调用包括文本、图片、音频和视频等在内的各种文件。
使用步骤:
Bucket:存储空间是用户用于存储对象(Object,就是文件)的容器,所有的对象都必须隶属于某个存储空间。
SDK的依赖(jar包)、代码示例等,都可以叫做SDK
使用服务进行文件上传
- AliOssUtil工具类
@Data
@AllArgsConstructor
@Slf4j
public class AliOssUtil {
private String endpoint;
private String accessKeyId;
private String accessKeySecret;
private String bucketName;
/**
* 文件上传
*
* @param bytes
* @param objectName
* @return
*/
public String upload(byte[] bytes, String objectName) {
// 创建OSSClient实例。
OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret);
try {
// 创建PutObject请求。
ossClient.putObject(bucketName, objectName, new ByteArrayInputStream(bytes));
} catch (OSSException oe) {
System.out.println("Caught an OSSException, which means your request made it to OSS, "
+ "but was rejected with an error response for some reason.");
System.out.println("Error Message:" + oe.getErrorMessage());
System.out.println("Error Code:" + oe.getErrorCode());
System.out.println("Request ID:" + oe.getRequestId());
System.out.println("Host ID:" + oe.getHostId());
} catch (ClientException ce) {
System.out.println("Caught an ClientException, which means the client encountered "
+ "a serious internal problem while trying to communicate with OSS, "
+ "such as not being able to access the network.");
System.out.println("Error Message:" + ce.getMessage());
} finally {
if (ossClient != null) {
ossClient.shutdown();
}
}
//文件访问路径规则 https://BucketName.Endpoint/ObjectName
StringBuilder stringBuilder = new StringBuilder("https://");
stringBuilder
.append(bucketName)
.append(".")
.append(endpoint)
.append("/")
.append(objectName);
log.info("文件上传到:{}", stringBuilder.toString());
return stringBuilder.toString();
}
}
- 在MVC中创建AliOssUtil第三方Bean,交给Spring容器管理
@Configuration
@Slf4j
public class OssConfiguration {
@Bean
@ConditionalOnMissingBean
public AliOssUtil aliOssUtil(AliOssProperties aliOssProperties){
log.info("开始创建阿里云文件上传工具类对象:{}",aliOssProperties);
return new AliOssUtil(aliOssProperties.getEndpoint(),
aliOssProperties.getAccessKeyId(),
aliOssProperties.getAccessKeySecret(),
aliOssProperties.getBucketName());
}
}
- 在Controller层写代码
@RestController
@RequestMapping("/admin/common")
@Slf4j
@Api(tags = "通用接口")
public class CommonController {
@Autowired
private AliOssUtil aliOssUtil;
@PostMapping("/upload")
@ApiOperation("上传文件接口")
public Result<String> upload(MultipartFile file){
log.info("文件上传:{}",file);
try {
//原始文件名
String originalFilename = file.getOriginalFilename();
//截取原始文件名的后缀 dfdfdf.png
String extension = originalFilename.substring(originalFilename.lastIndexOf("."));
//构造新文件名称
String objectName = UUID.randomUUID().toString().replace("-","") + extension;
//文件的请求路径
String filePath = aliOssUtil.upload(file.getBytes(), objectName);
return Result.success(filePath);
} catch (IOException e) {
log.error("文件上传失败:{}", e);
}
return Result.error(MessageConstant.UPLOAD_FAILED);
}
}
下面是请求参数的方法:
新增菜品
接口设计
代码书写
分页查询
接口设计
代码书写
写多表查询,并返回DishVO
批量删除
接口设计
代码书写
修改菜品
先进行数据回显
接口设计
代码书写
注:多表查询方式
多条sql方式查询:适合数据量大,互联网项目
一条sql方式查询(多表连接):适合数据量小,非互联网项目(企业内部使用的系统)
修改菜品
接口设计