2024黑马程序员 SpringCloud微服务开发与实战,Java黑马商城项目微服务实战开发(涵盖MybatisPlus、Docker、MQ、ES、Redis高级等)的个人学习心得与代码记录!Day1
SpringCloud微服务课程导学_哔哩哔哩_bilibili 官方文档 day01-MybatisPlus - 飞书云文档 (feishu.cn) 我这份文档是对官方文档的补充,比如官方文档没有记载P10的请求参数,这种东西没有什么营养,又容易漏写些东西,因此我记录了下来,您可以通过文字的目录快捷需要的东西 { “balance”: 2000, “info”: “{“age”:21}”, “password”: “123”, “phone”: “13899776876”, “username”: “WangWu” } 还记录了一些我自己开发中遇到的bug,如果您也遇到了可以直接地解决 不用去网上找了半天,也没能找到合适的解决方案 又比如说 P15 DB静态工具练习 没有给出修改后的代码 但我这里写了 还有一些我对技术点的理解
Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name ‘com.baomidou.mybatisplus.autoconfigure.MybatisPlusAutoConfiguration’: Unsatisfied dependency expressed through constructor parameter 0; nested exception is org.springframework.boot.context.properties.ConfigurationPropertiesBindException: Error creating bean with name ‘mybatis-plus-com.baomidou.mybatisplus.autoconfigure.MybatisPlusProperties’: Could not bind properties to ‘MybatisPlusProperties’ : prefix=mybatis-plus, ignoreInvalidFields=false, ignoreUnknownFields=true; nested exception is org.springframework.boot.context.properties.bind.BindException: Failed to bind properties under ‘mybatis-plus.global-config.db-config’ to com.baomidou.mybatisplus.core.config.GlobalConfig$DbConfig
mybatis-plus:
type-aliases-package: com.itheima.mp.domain.po
global-config:
db-config:
id-type: assign_id
QueryWrapper
是 MyBatis Plus 中的一个查询条件封装器,用于构建 SQL 查询条件。它提供了一种更加便捷和灵活的方式来构建复杂的查询条件,而不需要直接编写 SQL 语句。
通过 QueryWrapper
,你可以在 Java 代码中以面向对象的方式构建查询条件,而不必担心 SQL 注入等安全问题,同时也提高了代码的可读性和可维护性。
以下是 QueryWrapper
的一些常用方法和用法:
@Test
void testLambdaQueryWrapper() {
// 1.构建条件 WHERE username LIKE "%o%" AND balance >= 1000
QueryWrapper<User> wrapper = new QueryWrapper<>();
wrapper.lambda()
.select(User::getId, User::getUsername, User::getInfo, User::getBalance)
.like(User::getUsername, "o")
.ge(User::getBalance, 1000);
// 2.查询
List<User> users = userMapper.selectList(wrapper);
users.forEach(System.out::println);
}
这里代码写的很巧妙,代码的目的是将User Service交给Spring自动装配
@Autowired
private IUserService userService;
public UserController(IUserService userService) {
this.userService = userService;
}
但太繁琐了 于是采用
@Api(tags = "用户管理接口")
@RequiredArgsConstructor
@RestController
@RequestMapping("users")
public class UserController {
private final IUserService userService;
@RequiredArgsConstructor
注解,这是Lombok库提供的功能之一。它会为带有final
关键字的成员变量生成构造函数参数,并在构造函数中进行初始化。因此,在这种情况下,不需要使用@Autowired
注解来进行依赖注入,因为Lombok会自动为userService
生成一个构造函数参数,并且在初始化UserController
对象时将其注入。
@Autowired
,而且对于final
成员变量的使用更加规范,因为它们只能在构造函数中被初始化一次。
netstat -ano | findstr :8080
taskkill /PID 6176 /F
taskkill /PID 5776 /F
如果关掉后在运行关掉8080命令,那是有东西在一直重启通信,应该是你的苍穹外卖、黑马点评什么的没有关闭
打开任务管理器
诶个突突了
还不行就 这个是关闭所有niginx
taskkill /F /IM nginx.exe > nul
{
"balance": 2000,
"info": "{\"age\":21}",
"password": "123",
"phone": "13899776876",
"username": "WangWu"
}
control
server
@Override
public List<User> queryUsers(String name, Integer status, Integer minBalance, Integer maxBalance) {
return lambdaQuery()
.like(name != null, User::getUsername, name)
.eq(status != null, User::getStatus, status)
.ge(minBalance != null, User::getBalance, minBalance)
.le(maxBalance != null, User::getBalance, maxBalance)
.list();
}
@Test
void testSaveOneByOne() {
long b = System.currentTimeMillis();
for (int i = 1; i <= 100000; i++) {
userService.save(buildUser(i));
}
long e = System.currentTimeMillis();
System.out.println("耗时:" + (e - b));
}
// 下面这种数据能够提升10倍
@Test
void testSaveBatch() {
// 准备10万条数据
List<User> list = new ArrayList<>(1000);
long b = System.currentTimeMillis();
for (int i = 1; i <= 100000; i++) {
list.add(buildUser(i));
// 每1000条批量插入一次
if (i % 1000 == 0) {
userService.saveBatch(list);
list.clear();
}
}
long e = System.currentTimeMillis();
System.out.println("耗时:" + (e - b));
}
private User buildUser(int i) {
User user = new User();
user.setUsername("user_" + i);
user.setPassword("123");
user.setPhone("" + (18688190000L + i));
user.setBalance(2000);
user.setInfo("{\"age\": 24, \"intro\": \"英文老师\", \"gender\": \"female\"}");
user.setCreateTime(LocalDateTime.now());
user.setUpdateTime(user.getCreateTime());
return user;
}
在这两种处理方案中,第一个 testSaveOneByOne
方法是逐个保存每个用户的数据,而第二个 testSaveBatch
方法是批量保存用户数据。
下面是为什么批量保存会更快的一些原因:
批量保存数据可以减少数据库交互次数、减少事务开销,并且可以享受数据库的写入性能优化,因此通常会比逐个保存数据的方式更快。
MySQL的客户端连接参数中有这样的一个参数:rewriteBatchedStatements
。顾名思义,就是重写批处理的statement
语句。参考文档:
https://dev.mysql.com/doc/connector-j/8.0/en/connector-j-connp-props-performance-extensions.html#cj-conn-prop_rewriteBatchedStatements
这个参数的默认值是false,我们需要修改连接参数,将其配置为true
修改项目中的application.yml文件,在jdbc的url后面添加参数&rewriteBatchedStatements=true
:
spring:
datasource:
url: jdbc:mysql://127.0.0.1:3306/mp?useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&serverTimezone=Asia/Shanghai&rewriteBatchedStatements=true
driver-class-name: com.mysql.cj.jdbc.Driver
username: root
password: 123456
小蓝鸟怎么用 看我的另一篇文章
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.28</version> <!-- JDBC依赖 -->
</dependency>
jdbc:mysql://localhost:3306/mp?serverTimezone=Asia/Shanghai
依赖注入的核心思想是将类的依赖关系从类本身解耦,使得类更加灵活、可测试和可维护。
两个类相互注入可能会导致循环依赖的问题。循环依赖指的是两个或多个类之间存在直接或间接的依赖关系,使得它们无法被实例化。例如,类 A 依赖于类 B,而类 B 又依赖于类 A,这样的情况就会导致循环依赖。
循环依赖可能会导致以下问题:
为了避免循环依赖,可以考虑以下几种方法:
public void setAddresses(List<AddressVO> addressVOList) {
// 在这里将收货地址列表设置到用户VO中
// 例如,可以直接将地址列表赋值给用户VO中的地址属性
this.addressVOList = addressVOList;
}
@Override
public List<UserVO> queryUserAndAddressByIds(List<Long> ids) {
return List.of();
}
@Override
public List<UserVO> queryUserAndAddressByIds(List<Long> ids) {
// 1 查询用户
List<User> users = listByIds(ids);
if (CollUtil.isEmpty(users)) {
return Collections.emptyList();
}
// 2 查询地址 获取用户集合
List<Long> userIds = users.stream().map(User::getId).collect(Collectors.toList());
// 2.2 根据用户id查询地址
List<Address> addresses = Db.lambdaQuery(Address.class).in(Address::getUserId, userIds).list();
// 3 处理vo
List<UserVO> userVOS = BeanUtil.copyToList(users, UserVO.class);
userVOS.forEach(userVO -> {
List<AddressVO> addressVOS = BeanUtil.copyToList(
addresses.stream()
.filter(address -> address.getUserId().equals(userVO.getId()))
.collect(Collectors.toList()),
AddressVO.class);
userVO.setAddresses(addressVOS);
});
return userVOS;
}
package com.itheima.mp.enums;
import com.baomidou.mybatisplus.annotation.EnumValue;
import lombok.Getter;
@Getter
public enum UserStatus {
NORMAL(1, "正常"),
FREEZE(2, "冻结")
;
@EnumValue
private final int value;
private final String desc;
UserStatus(int value, String desc) {
this.value = value;
this.desc = desc;
}
}
@EnumValue
和 @JsonValue
是两个注解,通常用于在 Java 枚举类中定义枚举值的序列化和反序列化方式。
@EnumValue
注解标记的字段的值。这个值应该是枚举类中定义的一个整数字段,通常用来表示数据库中的实际存储值。@JsonValue
注解标记的字段的值作为 JSON 字符串的值。
从PageQuery
到MybatisPlus
的Page
之间转换的过程比较麻烦的