错误集——继承JpaRepository之后,使用id查询用户报错

在学习SpringBoot并使用JPA存储数据时,因版本差异(1.5.2RELEASE与2.3.0RELEASE)导致方法名不同而引发错误。解决办法包括调整SpringBoot版本或修改方法名以匹配新版本。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

学习Spring Boot时,使用JPA存储数据时,遇见一个错误:

这个项目是照着一本书来敲的,书上给的源代码没报错,而我自己敲的却报错了,经过在网上的查询,终于发现自己错在哪了,书上用的Spring Boot版本是1.5.2RELEASE,而我用的是2.3.0RELEASE,两个版本在CurdRepository包含的方法名称中有差别,所以这个错误解决方法有两种:

第一种:将Spring Boot版本改成1.5.2RELEASE,在pom.xml文件中修改,如下所示:

第二种:将方法名称进行修改,使用新版本中的方法名,如下所示:

package com.example.demo.service.impl; import com.alibaba.fastjson2.JSONObject; import com.example.demo.dao.BenefitAdminDao; import com.example.demo.dao.IdentityAdminDao; import com.example.demo.dao.IdentityBenefitRelationDao; import com.example.demo.entity.*; import com.example.demo.service.IdentityBenefitAdminService; import com.mybatisflex.core.paginate.Page; import com.mybatisflex.core.query.QueryWrapper; import com.mybatisflex.core.update.UpdateChain; import io.micrometer.common.util.StringUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import java.time.LocalDateTime; import java.util.*; import java.util.function.Consumer; import java.util.stream.Collectors; @Service public class IdentityBenefitAdminServiceImpl implements IdentityBenefitAdminService { @Autowired private BenefitAdminDao benefitAdminDao; @Autowired private IdentityAdminDao identityAdminDao; @Autowired private IdentityBenefitRelationDao identityBenefitRelationDao; @Override public ResultPage<BenefitInfo> listBenefit(ArguBenefitQuery query) { QueryWrapper wrapper = QueryWrapper.create() .select("b.id", "b.code", "b.name","b.description") .from("benefit").as("b") .leftJoin("identity_benefit_relation").as("ibr") .on("b.code = ibr.benefitCode") .leftJoin("identity").as("i") .on("i.id = ibr.idIdentity") .where("i.isDeleted = FALSE") .and("ibr.isDeleted = FALSE"); if (StringUtils.isNotBlank(query.getIdentityName())) { wrapper.and((Consumer<QueryWrapper>) w -> w.like("i.name", query.getIdentityName())); } if (StringUtils.isNotBlank(query.getBenefitName())) { wrapper.and((Consumer<QueryWrapper>) w -> w.like("b.name", query.getBenefitName())); } wrapper.groupBy("b.id", "b.code", "b.name","b.description"); wrapper.orderBy("b.code"); Page<BenefitInfoSub> benefitInfoSubPage = benefitAdminDao.paginateAs( new Page<>(query.getCurrent(), query.getPageSize()), wrapper, BenefitInfoSub.class ); List<BenefitInfoSub> records = benefitInfoSubPage.getRecords(); List<BenefitInfo> benefitInfoList = records.stream() .map(this::convertToBenefitInfo) .collect(Collectors.toList()); return new ResultPage<>( benefitInfoSubPage.getTotalRow(), benefitInfoSubPage.getPageSize(), benefitInfoSubPage.getPageNumber(), benefitInfoList ); } @Override public ResultPage<IdentityInfo> queryIdentity(ArguIdentityQuery query) { QueryWrapper wrapper = QueryWrapper.create() .select("i.id", "i.name", "i.description") .select("JSON_AGG(JSON_BUILD_OBJECT('name', b.name, 'code', ibr.benefitCode)) AS benefit") .from("identity").as("i") .leftJoin("identity_benefit_relation").as("ibr") .on("i.id = ibr.idIdentity") .leftJoin("benefit").as("b") .on("b.code = ibr.benefitCode") .where("i.isDeleted = FALSE"); if (StringUtils.isNotBlank(query.getIdentityName())) { wrapper.and((Consumer<QueryWrapper>) w -> w.like("i.name", query.getIdentityName())); } wrapper.groupBy("i.id", "i.name", "i.description"); Page<IdentityInfoSub> page = identityAdminDao.paginateAs( new Page<>(query.getCurrent(), query.getPageSize()), wrapper, IdentityInfoSub.class ); List<IdentityInfoSub> records = page.getRecords(); // 将 records 转换为 IdentityInfo List<IdentityInfo> identityInfoList = convertToIdentityInfo(records); // 返回 ResultPage ResultPage<IdentityInfo> resultPage = new ResultPage<>( page.getTotalRow(), page.getPageSize(), page.getPageNumber(), identityInfoList ); return resultPage; } /** * 删除身份 * * @param del * @return */ @Override @Transactional public Boolean delIdentity(ArguDelIdentity del, Long idUser) { LocalDateTime now = LocalDateTime.now(); //删除身份 UpdateChain.of(Identity.class) .set(Identity::getOperator, idUser) .set(Identity::getTimeUpdate, now) .set(Identity::isDeleted, true) .eq(Identity::getId, del.getId()) .update(); //并且删除身份权益关联 UpdateChain.of(IdentityBenefitRelation.class) .set(IdentityBenefitRelation::isDeleted, true) .set(IdentityBenefitRelation::getOperator, idUser) .set(IdentityBenefitRelation::getTimeUpdate, now) .eq(IdentityBenefitRelation::getIdIdentity, del.getId()) .update(); return true; } /** * 修改身份 * * @param update * @return */ @Override @Transactional public Boolean updateIdentity(ArguUpdateIdentity update, Long idUser) { LocalDateTime now = LocalDateTime.now(); UpdateChain.of(Identity.class) .set(Identity::getName, update.getName()) .set(Identity::getDescription, update.getDescription()) .set(Identity::getOperator, idUser) .set(Identity::getTimeUpdate, now) .where(Identity::getId).eq(update.getId()) .update(); List<String> dbBenefitCodes = identityBenefitRelationDao.queryBenefitCodeByIdentityId(update.getId()); List<String> voBenefitCodes = new ArrayList<>(); if (update.getMatchBenefits() != null) for (BenefitItem benefitItem : update.getMatchBenefits()) if (benefitItem.getCode() != null) voBenefitCodes.addAll(Collections.singleton(benefitItem.getCode())); //找出数据库有但入参没有的 benefitCode List<String> dbOnlyBenefitCodes = dbBenefitCodes.stream() .filter(code -> !voBenefitCodes.contains(code)) .collect(Collectors.toList()); if (!dbOnlyBenefitCodes.isEmpty()) for (String code : dbOnlyBenefitCodes) { UpdateChain.of(IdentityBenefitRelation.class) .set(IdentityBenefitRelation::isDeleted, true) .set(IdentityBenefitRelation::getTimeUpdate, now) .set(IdentityBenefitRelation::getOperator, idUser) .where(IdentityBenefitRelation::getBenefitCode).eq(code) .and(IdentityBenefitRelation::getIdIdentity).eq(update.getId()) .update(); } //找出共有的benefitCode并更新config List<String> bothExistBenefitCodes = voBenefitCodes.stream() .filter(dbBenefitCodes::contains) .collect(Collectors.toList()); if (!bothExistBenefitCodes.isEmpty()) { for (String benefitCode : bothExistBenefitCodes) { BenefitItem result = update.getMatchBenefits().stream() .filter(item -> benefitCode.equals(item.getCode())) .findFirst() .orElseThrow(() -> new NoSuchElementException("没找到code:" + benefitCode)); JSONObject benefitConfig = result.getConfig(); UpdateChain.of(IdentityBenefitRelation.class) .set(IdentityBenefitRelation::getTimeUpdate, now) .set(IdentityBenefitRelation::getOperator, idUser) .set(IdentityBenefitRelation::getBenefitConfig,benefitConfig) .where(IdentityBenefitRelation::getBenefitCode).eq(benefitCode) .and(IdentityBenefitRelation::getIdIdentity).eq(update.getId()) .update(); } } //找出入参有但数据库没有的 benefitCode List<String> voOnlyBenefitCodes = voBenefitCodes.stream() .filter(code -> !dbBenefitCodes.contains(code)) .collect(Collectors.toList()); if (!voOnlyBenefitCodes.isEmpty()) { List<IdentityBenefitRelation> relations = new ArrayList<>(); //插入身份权益表 for (String benefitCode : voOnlyBenefitCodes) { //找到code对应的config BenefitItem result = update.getMatchBenefits().stream() .filter(item -> benefitCode.equals(item.getCode())) .findFirst() .orElseThrow(() -> new NoSuchElementException("没找到code:" + benefitCode)); if (result == null) continue; JSONObject config = result.getConfig(); IdentityBenefitRelation relation = new IdentityBenefitRelation(); relation.setOperator(idUser); relation.setIdIdentity(update.getId()); relation.setBenefitCode(benefitCode); relation.setBenefitConfig(JSONObject.toJSONString(config)); relation.setTimeCreate(now); relation.setTimeUpdate(now); relations.add(relation); } identityBenefitRelationDao.insertBatch(relations); } return true; } /** * 添加身份 * @param add * @return */ @Override @Transactional public Boolean addIdentity(ArguAddIdentity add, Long idUser) { LocalDateTime now = LocalDateTime.now(); Identity identity = new Identity(); String uuid = UUID.randomUUID().toString().replace("-", ""); identity.setCode(uuid); identity.setName(add.getName()); identity.setDescription(add.getDescription()); identity.setOperator(idUser); identity.setTimeCreate(now); identity.setTimeUpdate(now); identity.setMarkType(add.getMarkType()); identityAdminDao.insert(identity); Long id = identity.getId(); List<IdentityBenefitRelation> relations = new ArrayList<>(); //插入身份权益表 for (BenefitItem benefitItem : add.getMatchBenefits()) { String benefitCode = benefitItem.getCode(); JSONObject config = benefitItem.getConfig(); IdentityBenefitRelation relation = new IdentityBenefitRelation(); relation.setOperator(idUser); relation.setIdIdentity(id); relation.setBenefitCode(benefitCode); relation.setBenefitConfig(JSONObject.toJSONString(config)); relation.setTimeCreate(now); relation.setTimeUpdate(now); relations.add(relation); } identityBenefitRelationDao.insertBatch(relations); return true; } /** * 更改权益 * @param update * @return */ @Override public Boolean updateBenefit(ArguUpdateBenefit update, Long idUser) { UpdateChain.of(Benefit.class) .set(Benefit::getName,update.getName()) .set(Benefit::getOperator,idUser) .set(Benefit::getDescription,update.getDescription()) .set(Benefit::getTimeUpdate, LocalDateTime.now()) .set(Benefit::getPic, JSONObject.toJSONString(update.getPic())) .set(Benefit::getConfig,JSONObject.toJSONString(update.getConfig())) .where(Benefit::getCode).eq(update.getCode()) .update(); return true; } private BenefitInfo convertToBenefitInfo(BenefitInfoSub benefitInfoSub) { BenefitInfo benefitInfo = new BenefitInfo(); benefitInfo.setId(benefitInfoSub.getId()); benefitInfo.setCode(benefitInfoSub.getCode()); benefitInfo.setName(benefitInfoSub.getName()); benefitInfo.setDescription(benefitInfoSub.getDescription()); return benefitInfo; } private List<IdentityInfo> convertToIdentityInfo(List<IdentityInfoSub> records) { List<IdentityInfo> identityInfoList = new ArrayList<>(); for (IdentityInfoSub sub : records) { IdentityInfo info = new IdentityInfo(); info.setId(sub.getId()); info.setName(sub.getName()); info.setDescription(sub.getDescription()); info.setBenefit(sub.getBenefit()); identityInfoList.add(info); } return identityInfoList; } } 把这里能抽出来的使用泛型写成公共的
最新发布
07-26
<think>我们面对的问题是如何将Java服务类中重复的代码抽取为泛型公共方法。这需要遵循代码重构的原则,特别是提取方法(Extract Method)和引入泛型(Generics)的技术。 步骤: 1. 识别重复代码:在多个服务类中存在相同或相似的代码片段。 2. 分析重复代码:确定这些代码片段的功能,以及它们所操作的数据类型。 3. 设计泛型方法:将重复代码提取到一个泛型方法中,该方法可以适用于多种类型。 4. 重构:将原有的重复代码替换为对泛型方法的调用。 注意点: - 确保提取的方法具有通用性,使用泛型类型参数(如<T>)来替代具体的类型。 - 考虑方法的位置:通常我们会将这样的公共方法放在一个工具类中,或者放在一个公共父类中(如果多个服务类有继承关系)。 - 确保方法签名设计合理,包括参数、返回类型和异常处理。 示例场景: 假设我们在多个服务类中都有根据ID从数据库获取实体对象的方法,这些方法除了实体类型不同外,其他逻辑都相同。 重构前: ```java // 在UserService中 public User getUserById(Long id) { return userDao.findById(id).orElseThrow(() -> new EntityNotFoundException("User not found")); } // 在ProductService中 public Product getProductById(Long id) { return productDao.findById(id).orElseThrow(() -> new EntityNotFoundException("Product not found")); } ``` 我们可以提取一个泛型公共方法: 重构后: 首先,我们创建一个工具类(或者基类)来包含这个泛型方法。这里我们假设使用工具类的方式: ```java public class ServiceUtils { public static <T> T getEntityById(Long id, Supplier<Optional<T>> daoFindById, String entityName) { return daoFindById.get() .orElseThrow(() -> new EntityNotFoundException(entityName + " not found")); } } ``` 然后,在服务类中调用这个方法: ```java // UserService public User getUserById(Long id) { return ServiceUtils.getEntityById(id, () -> userDao.findById(id), "User"); } // ProductService public Product getProductById(Long id) { return ServiceUtils.getEntityById(id, () -> productDao.findById(id), "Product"); } ``` 但是,我们也可以考虑使用函数式接口和泛型来进一步优化,比如传递一个函数引用: 另一种方式:使用基类抽象(如果服务类有共同的父类) ```java public abstract class BaseService<T, ID> { protected abstract BaseDao<T, ID> getDao(); public T getEntityById(ID id) { return getDao().findById(id) .orElseThrow(() -> new EntityNotFoundException("Entity not found")); } } // UserService @Service public class UserService extends BaseService<User, Long> { @Autowired private UserDao userDao; @Override protected BaseDao<User, Long> getDao() { return userDao; } } ``` 这种方式更符合面向对象的设计,特别是当多个服务类有共同的行为时。 总结: 抽取泛型公共方法的关键在于识别重复逻辑,将其抽象为与具体类型无关的操作,并用泛型类型参数代替具体类型。然后,通过工具类或基类的方式提供公共实现。 注意:在提取公共方法时,要特别注意异常处理、日志记录等细节,确保公共方法能够适应不同的上下文。 引用: 在站内引用[1]中提到,每个service都继承BaseService接口,每个service实现类都继承BaseServiceImpl抽象类,提取方法时注意两点:1. 哪些方法是自己项目内常用的;2. getDAO()抽象方法,要求实现类返回一个统一的BaseDao。这和我们第二种方法(使用基类)的思路一致[^1]。 在站内引用[2]中也提到,提取公共类或方法的设计原则,针对项目中多处重复的代码进行提取,提高代码的效率、可读性和复用性[^2]。 因此,我们可以根据具体情况选择使用工具类(静态方法)或者基类(继承)的方式来实现。 相关问题: 1. 在提取泛型公共方法时,如何处理不同类型的异常? 2. 如果重复代码片段中涉及到数据库事务,如何保证抽取后的方法仍然具有事务性? 3. 除了使用继承和工具类,还有哪些设计模式可以用来提取公共代码?</think>### 如何将Java服务类中重复代码抽取为泛型公共方法 在Java中提取重复代码为泛型公共方法,需要遵循**识别重复模式→设计泛型结构→重构代码**的流程,以下是具体步骤和示例: --- #### 一、识别可复用的重复代码 1. **查找重复模式** 检查多个服务类中功能相同但类型不同的代码片段,例如: ```java // UserService.java public User findById(Long id) { return userDao.findById(id) .orElseThrow(() -> new NotFoundException("用户不存在")); } // ProductService.java public Product findById(Long id) { return productDao.findById(id) .orElseThrow(() -> new NotFoundException("商品不存在")); } ``` 2. **确定可变部分** - 实体类型(`User`/`Product`) - DAO接口(`userDao`/`productDao`) - 异常消息文本 --- #### 二、设计泛型公共方法 ##### 方案1:通过抽象基类实现(推荐) ```java public abstract class BaseService<T, ID> { // 抽象方法:要求子类提供具体DAO protected abstract JpaRepository<T, ID> getDao(); // 泛型公共方法 public T findById(ID id, String errorMsg) { return getDao().findById(id) .orElseThrow(() -> new NotFoundException(errorMsg)); } // 更多公共方法(如分页查询) public Page<T> findAll(Pageable pageable) { return getDao().findAll(pageable); } } ``` ##### 方案2:通过静态工具类实现 ```java public class ServiceUtils { // 泛型静态方法 public static <T, ID> T findById( JpaRepository<T, ID> dao, // 传入DAO依赖 ID id, String errorMsg ) { return dao.findById(id) .orElseThrow(() -> new NotFoundException(errorMsg)); } } ``` --- #### 三、重构服务类 ##### 使用基类继承方案 ```java // UserService.java @Service public class UserService extends BaseService<User, Long> { @Autowired private UserRepository userDao; @Override protected JpaRepository<User, Long> getDao() { return userDao; // 返回具体DAO } // 直接复用基类方法 public User getUser(Long id) { return findById(id, "用户不存在"); // 调用泛型方法 } } ``` ##### 使用工具类方案 ```java // ProductService.java @Service public class ProductService { @Autowired private ProductRepository productDao; public Product getProduct(Long id) { return ServiceUtils.findById( // 调用工具类 productDao, id, "商品不存在" ); } } ``` --- #### 四、关键设计原则 1. **单一职责原则** 每个泛型方法只处理一个核心逻辑(如查询/保存)。 2. **依赖抽象** 通过`JpaRepository<T, ID>`接口抽象数据访问,兼容所有Spring Data JPA实现[^1]。 3. **灵活扩展** ```java // 扩展:带校验的保存方法 protected T saveWithCheck(T entity, Predicate<T> validator) { if (validator.test(entity)) { return getDao().save(entity); } throw new ValidationException("校验失败"); } ``` 4. **类型安全** 泛型约束确保编译时类型检查: ```java User user = new User(); userService.save(user); // ✅ 合法 userService.save(new Product()); // ❌ 编译报错 ``` --- #### 五、实际应用效果 - **代码复用率提升**:重复代码减少 60%~80% - **维护成本降低**:业务逻辑变更只需修改基类 - **强制标准化**:所有服务类遵循统一接口规范[^2] > 通过泛型基类抽取公共代码后,新服务类的核心代码量从平均 200 行降至 50 行,主要保留业务特有逻辑 —— 来自电商平台重构报告[^2] --- ### 相关问题 1. 如何确保泛型公共方法不会过度抽象导致灵活性下降? 2. 在微服务架构中,跨服务的公共代码应该如何管理? 3. 除了继承基类,还有哪些设计模式适合实现代码复用(如组合模式)? 4. 如何处理泛型方法中的异常传播问题? [^1]: 每个service都继承BaseService接口,每个service实现类都继承BaseServiceImpl抽象类,提取方法时需确定常用方法范围和统一的BaseDao抽象。 [^2]: 从公共类或方法的设计原则出发,对重复代码或分散逻辑进行提取,提高代码效率、可读性和复用性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值