上班笔记之java自带注解(Javax)的@PostConstruct,用于缓存数据。list<实体类>的按实体类的属性分组在Map集合中,list<实体类>按照实体类的属性像SQL语句一样进行过滤。

本文介绍了Java的@PostConstruct注解,澄清了它并非仅属于Spring框架,而是Java自身的一部分。@PostConstruct注解用于标记在服务器加载Servlet时执行的非静态void方法,且仅执行一次,介于构造函数和init方法之间。在Spring中,其执行顺序为构造方法 -> @Autowired依赖注入 -> @PostConstruct注解的方法。@PostConstruct常用于缓存数据,如将数据库查询结果按特定属性分组存储在Map中。此外,文章还探讨了如何在静态方法中调用依赖注入的Bean方法,以及如何实现类似`where xx=null and yy=xx`的过滤逻辑在Java中实现。

@PostConstruct解释

@PostConstruct注解好多人以为是Spring提供的。其实是Java自己的注解。

Java中该注解的说明:@PostConstruct该注解被用来修饰一个非静态的void()方法。被@PostConstruct修饰的方法会在服务器加载Servlet的时候运行,并且只会被服务器执行一次。PostConstruct在构造函数之后执行,init()方法之前执行。

通常我们会是在Spring框架中使用到@PostConstruct注解 该注解的方法在整个Bean初始化中的执行顺序:

Constructor(构造方法) -> @Autowired(依赖注入) -> @PostConstruct(注释的方法)

应用:在静态方法中调用依赖注入的Bean中的方法。

@PostConstruct,在缓存数据时用,把数据库查询的数据放在Map<String,list<实体类>>中,string是实体类中的一个属性,用于把实体类分类分组,用于快速搜索

   **  第一次根据keyname的value值创建一个新arraylist,把这次的keyname作为map的string,并把空的list作为value,再把dto添加到list。二次,再根据keyname的value,如果keyname已经出现过一次,不在new新的list,直接dto add到list中,这样循环,相同的keyname的dto就会全部聚集到第次new的list中,完成了按keyname的分组。**

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述在这里插入图片描述

sql的where xx=null and yy =“xx” 的过滤在java中实现在这里插入图片描述

package com.kucun.Service; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.ApplicationContext; import org.springframework.core.ResolvableType; import org.springframework.data.jpa.domain.Specification; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Service; import javax.annotation.PostConstruct; import java.util.Collections; import java.util.Date; import java.util.HashMap; import java.util.List; import java.util.Map; @Service public class DynamicRepositoryService { @Autowired private ApplicationContext applicationContext; private final Map<String, JpaRepository<?, ?>> repositoryMap = new HashMap<>(); private final Map<Class<?>, JpaRepository<?, ?>> classRepositoryMap = new HashMap<>(); private final Map<String, Class<?>> stringClassMap=new HashMap<>(); public Map<String, Class<?>> getStringClassMap() { return stringClassMap; } public Map<Class<?>, JpaRepository<?, ?>> getClassRepositoryMap() { return classRepositoryMap; } @PostConstruct public void init() { Map<String, JpaRepository> repositories = applicationContext.getBeansOfType(JpaRepository.class); repositories.forEach((beanName, repo) -> { Class<?> entityClass = resolveEntityClass(repo); if (entityClass != null) { repositoryMap.put(entityClass.getSimpleName().toLowerCase(), repo); classRepositoryMap.put(entityClass, repo); stringClassMap.put(entityClass.getSimpleName().toLowerCase(), entityClass); } }); } // 在DynamicRepositoryService中添加以下方法 public List<?> findUpdatedSince(String entityName, Date since) { try { Class<?> entityClass = stringClassMap.get(entityName); JpaRepository<?, ?> repo = repositoryMap.get(entityName); // 使用Criteria API构建查询 return repo.findUpdatedSince(since); } catch (Exception e) { return Collections.emptyList(); } } /** * 使用 ResolvableType 正确解析代理对象的泛型参数 */ private Class<?> resolveEntityClass(JpaRepository<?, ?> repository) { // 创建 ResolvableType 从 Repository 实例 ResolvableType repoType = ResolvableType.forClass(repository.getClass()); // 沿着继承链向上查找 JpaRepository 接口 ResolvableType jpaRepositoryType = repoType.as(JpaRepository.class); // 获取第一个泛型参数(实体类型) ResolvableType entityType = jpaRepositoryType.getGeneric(0); // 解析为 Class 对象 return entityType.resolve(); } public JpaRepository<?, ?> getJpaRepository(String entityName) { JpaRepository<?, ?> repo = repositoryMap.get(entityName.toLowerCase()); if (repo == null) { throw new IllegalArgumentException("未找到实 " + entityName + " 对应的Repository"); } return repo; } public JpaRepository<?, ?> getJpaRepository(Class<?> entityClass) { JpaRepository<?, ?> repo = classRepositoryMap.get(entityClass); if (repo == null) { throw new IllegalArgumentException("未找到实体类 " + entityClass.getName() + " 对应的Repository"); } return repo; } public Map<String, JpaRepository<?, ?>> getRepositoryNameMap() { return repositoryMap; } } 改造获取的是CustomRepository接口,所有的Repository都实现了CustomRepository接口 public interface CustomRepository<T, ID> extends JpaRepository<T, ID>{ List findUpdatedSince(Date active); // 符合命名规范 } @Repository public interface ChanpinZujianRepository extends CustomRepository<Chanpin_zujian, Integer> { boolean existsByChanpin_IdAndZujian_Id(Integer chanpinid, Integer zujianid); Chanpin_zujian findByChanpin_IdAndZujian_Id(Integer chanpinid, Integer zujianid); }
06-30
package com.kucun.Service; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.ApplicationContext; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.stereotype.Service; import javax.annotation.PostConstruct; import java.lang.reflect.ParameterizedType; import java.util.Arrays; import java.util.HashMap; import java.util.Map; @Service public class DynamicRepositoryService { @Autowired private ApplicationContext applicationContext; // 存储实名称(小写)到Repository的映射 private final Map<String, JpaRepository<?, ?>> repositoryMap = new HashMap<>(); // 存储实体类到Repository的映射 private final Map<Class<?>, JpaRepository<?, ?>> classRepositoryMap = new HashMap<>(); public Map<Class<?>, JpaRepository<?, ?>> getClassRepositoryMap() { return classRepositoryMap; } public DynamicRepositoryService() { super(); } /** * 初始化方法:扫描所有JpaRepository并建立映射 */ @PostConstruct public void init() { // 获取所有JpaRepository的实现 Map<String, JpaRepository> repositories = applicationContext.getBeansOfType(JpaRepository.class); repositories.forEach((beanName, repo) -> { // 通过反射获取Repository管理的实体类型 Class<?> entityClass = resolveEntityClass(repo); if (entityClass != null) { // 建立映射:实体类名(小写)-> Repository repositoryMap.put(entityClass.getSimpleName().toLowerCase(), repo); // 建立映射:实体类-> Repository classRepositoryMap.put(entityClass, repo); } }); } /** * 解析Repository管理的实体类 * @param repository JpaRepository实例 * @return 管理的实体类 */ private Class<?> resolveEntityClass(JpaRepository<?, ?> repository) { // 获取Repository接口的泛型参数 return Arrays.stream(repository.getClass().getGenericInterfaces()) .filter(genericInterface -> genericInterface.getTypeName().contains("JpaRepository")) .findFirst() .map(ParameterizedType.class::cast) .map(type -> (Class<?>) type.getActualTypeArguments()[0]) .orElse(null); } /** * 根据实名称获取Repository * @param entityName 实名称(小写) * @return 对应的JpaRepository */ public JpaRepository<?, ?> getJpaRepository(String entityName) { JpaRepository<?, ?> repo = repositoryMap.get(entityName.toLowerCase()); if (repo == null) { throw new IllegalArgumentException("未找到实 " + entityName + " 对应的Repository"); } return repo; } /** * 根据实体类获取Repository * @param entityClass 实体类 * @return 对应的JpaRepository */ public JpaRepository<?, ?> getJpaRepository(Class<?> entityClass) { JpaRepository<?, ?> repo = classRepositoryMap.get(entityClass); if (repo == null) { throw new IllegalArgumentException("未找到实体类 " + entityClass.getName() + " 对应的Repository"); } return repo; } /** * 获取所有注册的Repository映射 * @return 实名称到Repository的映射 */ public Map<String, JpaRepository<?, ?>> getRepositoryNameMap() { return repositoryMap; } } 两个集合是空的
06-21
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值