Ошибка создания bean-компонента с именем redisConnectionFactory.

avatar
user1398291
8 августа 2021 в 22:37
852
1
2

Мы используем RedisTemplate для вызовов Redis.

 @Bean
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {
    RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();
    redisTemplate.setConnectionFactory(redisConnectionFactory);
    redisTemplate.setEnableTransactionSupport(true);
    // Replace the default serialization with Jackson2JsonRedisSerialize
    Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);

    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
    objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);

    jackson2JsonRedisSerializer.setObjectMapper(objectMapper);

    // Set the value of the serialization rules and key serialization rules
    redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
    redisTemplate.setKeySerializer(new StringRedisSerializer());

    redisTemplate.setHashKeySerializer(jackson2JsonRedisSerializer);
    redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);

    redisTemplate.setDefaultSerializer(jackson2JsonRedisSerializer);
    redisTemplate.setEnableDefaultSerializer(true);
    redisTemplate.afterPropertiesSet();
    return redisTemplate;
}

Мы совершаем около 1K вызовов в минуту, например. redisTemplate.opsForHash().get(HASH_KEY, ключ);

мы получаем следующую ошибку:

stack_trace: org.springframework.beans.factory.BeanCreationNotAllowedException: Error creating bean with name 'redisConnectionFactory': Singleton bean creation not allowed while singletons of this factory are in destruction (Do not request a bean from a BeanFactory in a destroy method implementation!)
at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:212)
at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:318)
at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:199)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBeansOfType(DefaultListableBeanFactory.java:611)
at org.springframework.beans.factory.BeanFactoryUtils.beansOfTypeIncludingAncestors(BeanFactoryUtils.java:340)
at org.springframework.dao.support.PersistenceExceptionTranslationInterceptor.detectPersistenceExceptionTranslators(PersistenceExceptionTranslationInterceptor.java:168)
at org.springframework.dao.support.PersistenceExceptionTranslationInterceptor.invoke(PersistenceExceptionTranslationInterceptor.java:150)
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:186)
at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:691)

Мы используем SpringBoot + Redis 2.8.24 + JDK11.

Есть входные данные?

Источник
roookeee
12 августа 2021 в 23:07
1

Ошибка кажется не связанной с вашим "1k вызовов в минуту" - можете немного уточнить? В качестве нит: вы не должны вызывать afterPropertiesSet() вручную, Spring сделает это автоматически для всех объектов @Component / @Bean. Ошибка указывает на то, что приложение вообще не запускается, а не после определенного количества вызовов.

user1398291
13 августа 2021 в 18:51
0

Причина, по которой я упомянул о 1K, заключается в том, чтобы дать представление о трафике. Не все вызовы периодически терпят неудачу, мы получаем эту ошибку. Одно сомнение заключалось в том, есть ли какие-либо ограничения на пул соединений, которые могут вызывать эту ошибку.

Ответы (1)

avatar
Midhun Mohan
12 августа 2021 в 17:01
3

Вам может подойти следующий способ.

@Configuration
public class RedisConfig {

    @Value("${spring.redis.host}")
    private String host;
    @Value("${spring.redis.port}")
    private Integer port;
    @Value("${spring.redis.database}")
    private Integer database;

    @Bean
    JedisConnectionFactory jedisConnectionFactory() {
        RedisStandaloneConfiguration redisStandaloneConfiguration = new RedisStandaloneConfiguration(host, port);
        redisStandaloneConfiguration.setDatabase(database);
        return new JedisConnectionFactory(redisStandaloneConfiguration);
    }

    @Bean
    public RedisTemplate<String, Object> redisTemplate(JedisConnectionFactory jedisConnectionFactory) {
        RedisTemplate<String, Object> template = new RedisTemplate<>();
        template.setConnectionFactory(jedisConnectionFactory);
        template.setKeySerializer(new StringRedisSerializer());
        template.setValueSerializer(new StringRedisSerializer());
        template.setHashKeySerializer(new GenericToStringSerializer<>(String.class));
        template.setHashValueSerializer(new GenericToStringSerializer<>(String.class));
        return template;
    }
}

Вы можете установить требуемую сериализацию и получить доступ к данным на ее основе.

public class RedisHashRepository {

    final Logger logger = LoggerFactory.getLogger(RedisHashRepository.class);

    private HashOperations hashOperations;

    @Autowired
    public RedisHashRepository(RedisTemplate redisTemplate) {
        this.hashOperations = redisTemplate.opsForHash();
    }

    public String get(String mainKey,String key) {
        return (String) hashOperations.get(mainKey, key);
    }
}