Spring Boot JPA save之怪异

文章讲述了在SpringBoot项目中,使用JPA的save方法保存实体时遇到数据库异常(如主键冲突)无法捕获的问题,解决办法是通过使用saveAndFlush方法,确保异常在SQL执行后立即抛出并被捕获。
摘要由CSDN通过智能技术生成

刚刚开通了一个公众号,会分享一些技术博客和自己觉得比较好的项目,同时会更新一些自己使用的工具和图书资料,后面会整理一些面试资料进行分享,觉得有兴趣的可以关注一下。
在这里插入图片描述

项目场景

工作中的一个小需求,由于这个项目第一阶段只是引入,作为触发程序在后台跑数据,将下游的数据引入即可。所以不需要暴露在页面上。所以报错的话,是没有任何人感知的。所以需要提供一个监控程序,就是在遇到异常就记录下来,每日满足发生5次异常就发送邮件通知,通知后就不再通知。


问题描述

程序其实很简单,但是测试Demo的时候,由于使用的是JPA,使用的save方法保存,方法内部处理就是先判断isNew,然后去做update或者insert操作。问题是这个方法遇到数据库异常,比如主键冲突,虽然打印日志,但是异常实际上是被某个异常处理器处理掉了,根本报不出来。

try{
    dao.save(entity);
}catch (Exception e) {
    log.error("error happened", e);
    monitorExceptionService.recordError(e);
}

这个错误日志是进入不到catch里面的。


问题原因

网上资料介绍要使用saveAndFlush方法,查看内部源码,其实就是多了一个flush方法。flush其实就是将sql预编译执行的结果发送给数据库得到的结果。save方法是在sql统一提交的才会报错,就是@Transactional整体结束的时候报错,所以catch不到报错。而saveAndFlush方法多的flush方法就是将sql结果发送到数据库,所以立马能得到报错。

源码如下:

	@Transactional
	@Override
	public <S extends T> S saveAndFlush(S entity) {

		S result = save(entity);
		flush();

		return result;
	}

解决方案

就是将save方法改为saveAndFlush方法即可。
代码如下:

try{
    dao.saveAndFlush(entity);
}catch (Exception e) {
    log.error("error happened", e);
    monitorExceptionService.recordError(e);
}

搞定收工!

  • 6
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,针对你的问题,我可以为你提供一些思路和建议。 首先,你需要明确购物车功能的需求和实现方式。购物车功能通常需要记录用户选择的商品信息、数量、价格等,并提供添加、修改、删除购物车商品等操作。 在技术实现方面,你可以选择使用 Spring Boot 框架搭建项目,并集成 MariaDB 数据库和 JPA 框架实现数据持久化。具体步骤可以参考以下流程: 1. 创建 Spring Boot 项目,并添加相关依赖,如 Spring Boot Web、Spring Data JPA、MariaDB 驱动等。 2. 创建商品和购物车实体类,并使用 JPA 注解定义实体属性和关联关系。如: 商品实体类: ``` @Entity @Table(name = "product") public class Product { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id; @Column(nullable = false) private String name; @Column(nullable = false) private BigDecimal price; // 省略其他属性和方法 } ``` 购物车实体类: ``` @Entity @Table(name = "cart") public class Cart { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id; @ManyToOne @JoinColumn(name = "user_id", nullable = false) private User user; @ManyToOne @JoinColumn(name = "product_id", nullable = false) private Product product; @Column(nullable = false) private Integer quantity; // 省略其他属性和方法 } ``` 3. 创建用户实体类,并添加购物车属性和关联关系。如: 用户实体类: ``` @Entity @Table(name = "user") public class User { @Id @GeneratedValue(strategy = GenerationType.IDENTITY) private Long id; @Column(nullable = false) private String username; @OneToMany(mappedBy = "user") private List<Cart> carts = new ArrayList<>(); // 省略其他属性和方法 } ``` 4. 创建商品和购物车的 Repository 接口,并继承 JPA 的 CrudRepository 接口实现数据访问和操作。如: 商品 Repository 接口: ``` public interface ProductRepository extends CrudRepository<Product, Long> { // 省略自定义方法 } ``` 购物车 Repository 接口: ``` public interface CartRepository extends CrudRepository<Cart, Long> { List<Cart> findByUser(User user); } ``` 5. 创建商品和购物车的 Service 类,并注入对应的 Repository 实现业务逻辑。如: 商品 Service 类: ``` @Service public class ProductService { @Autowired private ProductRepository productRepository; // 省略自定义方法 } ``` 购物车 Service 类: ``` @Service public class CartService { @Autowired private CartRepository cartRepository; // 省略自定义方法 } ``` 6. 创建商品和购物车的 Controller 类,并注入对应的 Service 处理 HTTP 请求和响应。如: 商品 Controller 类: ``` @RestController @RequestMapping("/api/products") public class ProductController { @Autowired private ProductService productService; // 省略 HTTP 方法和请求处理 } ``` 购物车 Controller 类: ``` @RestController @RequestMapping("/api/carts") public class CartController { @Autowired private CartService cartService; // 省略 HTTP 方法和请求处理 } ``` 7. 最后,你可以实现购物车功能的具体操作,如添加商品到购物车、修改购物车商品数量、删除购物车商品等。具体实现可以参考以下代码: 添加商品到购物车: ``` @PostMapping("/{productId}/add") public void addToCart(@PathVariable Long productId, @RequestBody Integer quantity, Principal principal) { String username = principal.getName(); User user = userService.findByUsername(username); Product product = productService.findById(productId); Cart cart = cartService.findByUserAndProduct(user, product); if (cart != null) { cart.setQuantity(cart.getQuantity() + quantity); } else { cart = new Cart(); cart.setUser(user); cart.setProduct(product); cart.setQuantity(quantity); } cartService.save(cart); } ``` 修改购物车商品数量: ``` @PutMapping("/{cartId}") public void updateCart(@PathVariable Long cartId, @RequestBody Integer quantity) { Cart cart = cartService.findById(cartId); cart.setQuantity(quantity); cartService.save(cart); } ``` 删除购物车商品: ``` @DeleteMapping("/{cartId}") public void deleteCart(@PathVariable Long cartId) { cartService.deleteById(cartId); } ``` 以上是一个简单的购物车功能的实现思路和代码示例,你可以根据实际需求进行修改和扩展。希望能对你有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

梦幻D开始

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值