java - IgniteCache 不调用 loadCache()

标签 java spring caching ignite

我关注了ignite documentation并创建了我的商店适配器:

import java.io.Serializable;
import java.util.Map;
import java.util.Optional;
import javax.cache.Cache;
import javax.cache.integration.CacheLoaderException;
import javax.cache.integration.CacheWriterException;

import org.apache.ignite.Ignite;
import org.apache.ignite.cache.store.CacheStoreAdapter;
import org.apache.ignite.lang.IgniteBiInClosure;
import org.apache.ignite.resources.IgniteInstanceResource;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import com.my.calendar.dao.jpa.ChannelDetailsJpaRepository;
import com.my.calendar.entity.ChannelDetails;

@Service
public class CacheChannelStoreAdapter extends CacheStoreAdapter<Long, ChannelDetails> implements Serializable {
    private static final Logger LOGGER = LogManager.getLogger(CacheChannelStoreAdapter.class);

    private static ChannelDetailsJpaRepository channelDetailsJpaRepository;

    // Will be automatically injected. !!! apacheignite info, but it DOES NOT
    @IgniteInstanceResource
    private Ignite ignite;

    public Ignite getIgnite() {
        return ignite;
    }

    public void setIgnite(Ignite ignite) {
        this.ignite = ignite;
    }

    @Autowired
    public CacheChannelStoreAdapter(ChannelDetailsJpaRepository channelDetailsJpaRepository) {
        this.channelDetailsJpaRepository = channelDetailsJpaRepository;
    }

    @Override
    public void loadCache(IgniteBiInClosure<Long, ChannelDetails> clo, Object... args) {
        LOGGER.trace("load cache...");
        super.loadCache(clo, args);
    }

    @Override
    public Map<Long, ChannelDetails> loadAll(Iterable<? extends Long> keys) {
        LOGGER.trace("load all. keys: {}", keys);
        return super.loadAll(keys);
    }

    @Override
    public ChannelDetails load(Long key) throws CacheLoaderException {
        LOGGER.debug("object to load: {}", key);
        return channelDetailsJpaRepository.findOne(key);
    }

    @Override
    public void write(Cache.Entry<? extends Long, ? extends ChannelDetails> entry) throws CacheWriterException {
        LOGGER.debug("object to save: {}", entry);
        Optional.ofNullable(entry)
                .map(Cache.Entry::getValue)
                .ifPresent(channelDetailsJpaRepository::save);
    }

    @Override
    public void delete(Object key) throws CacheWriterException {
        LOGGER.debug("object to delete: {}", key);
        Optional.ofNullable(key)
                .filter(Long.class::isInstance)
                .map(Long.class::cast )
                .ifPresent(channelDetailsJpaRepository::delete);
    }
}

缓存适配器已注入(inject)缓存配置 cacheConfiguration.setCacheStoreFactory(new FactoryBuilder.SingletonFactory<>(cacheStoreAdapter));


我同时遇到了几个问题:

  • 没有看到 CacheStore.loadCache() 的调用| IgniteCache.loadCache() (应调用以下文档@Override public ChannelDetails load(Long key))
  • 没有看到 ignite 注入(inject) @IgniteInstanceResource private Ignite ignite;

有什么修复或检查的想法吗? ...也许我错过了重要的部分,我应该在 startTime 或其他地方直接调用缓存加载?

添加更多详细信息

所有缓存的数据都会在缓存时保存到我的数据库中。在服务器启动服务器时,没有从数据库加载数据。

我的适配器的缓存配置部分:

cacheConfiguration.setCacheStoreFactory(new FactoryBuilder.SingletonFactory<>(cacheStoreAdapter));
// Configure cache to use store.
cacheConfiguration.setReadThrough(true);
cacheConfiguration.setWriteThrough(true);
// Enable database batching.
cacheConfiguration.setWriteBehindEnabled(true);

Ignite 从 Web 服务器开始(在相同的 jvm 中,我知道它很难看,但它不能是我测试失败的原因)。

最佳答案

感谢尼古拉的建议。他没有提供代码,所以我认为知道我的情况是如何解决的会很好。我已经有了完整的解决方案。

Ignite配置部分我们应该调用

igniteConcreteCache.loadCache(
    (IgniteBiPredicate) (key, val) -> {
        System.out.println(" =========>>>> Loading [key=" + key + ", val=" + val + ']');
        return true;
    });

要从存储中加载数据,我们需要重写StoreAdapter中的相应方法:

@Override
public void loadCache(IgniteBiInClosure<Long, ChannelDetails> clo, Object... args) {
    LOGGER.trace("load cache...");
    super.loadCache(clo, args);
}

应该是:

@Override
public void loadCache(IgniteBiInClosure<Long, ChannelDetails> clo, Object... args) {
    LOGGER.trace("Load all cache data from db. this part should be used if Ignite starts.");
    if (args == null || args.length == 0) {
        channelDetailsJpaRepository.findAll()
                .forEach(channelDetails -> clo.apply(channelDetails.getId(), channelDetails));
    } else {
        throw new NotImplementedException("Handle to load objects by id collection.");
    }
}

这个解决方案并不完美(它还有很多地方需要做得更好),但它有效,我们可以使用它。

关于java - IgniteCache 不调用 loadCache(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44542172/

相关文章:

caching - jboss/tomcat tld 缓存失效

mysql - Django 上的 SQL 缓存

c# - Entity Framework ——如何缓存和共享只读对象

java - 如何找到两个给定范围之间的重叠?

java - 我想使用保存的首选项获取选定的单选按钮,但 Activity 没有打开?

java - 预检响应具有 oauth/token 的无效 HTTP 状态代码 401

条件约束的 Spring 验证

java - HikariCP 池连接不足

Java Regex - 在字符串中查找特定字符串

java - String.split() 的副作用,还是正则表达式错误?