Spring Data Redis:Redis管道返回始终为null

时间:2017-09-20 13:12:07

标签: java lua redis spring-data pipelining

我想只使用指定的字段检索多个hashmap值。所以我选择加入Redis管道。

在测试以下代码时,我看到redisResponse1始终为空,其中redisResponse2具有值。

    getRedisTemplate().executePipelined(new RedisCallback<Object>() { 
        @Override
        public Object doInRedis(RedisConnection connection) throws DataAccessException {
                List<byte[]> redisResponse1 = connection.hMGet(key.getBytes(), params);
                List<byte[]> redisResponse2 = getRedisTemplate().getConnectionFactory().getConnection().hMGet(key.getBytes(), specificParams);
                return null;
        }
    });

当我查看代码并发现下面的地方

a)redisResponse2未使用管道选项执行。

b)redisResponse1使用管道执行(isPipelined()== true)但返回始终为null。

public List<byte[]> hMGet(byte[] key, byte[]... fields) {
    try {
        if (isPipelined()) {
            pipeline(new JedisResult(pipeline.hmget(key, fields)));
            return null;
        }
        if (isQueueing()) {
            transaction(new JedisResult(transaction.hmget(key, fields)));
            return null;
        }
        return jedis.hmget(key, fields);
    } catch (Exception ex) {
        throw convertJedisAccessException(ex);
    }
}

所以问题是

1)如何通过管道选项实现我的用例?

2)在此RedisCallback中访问getRedisTemplate().getConnectionFactory().getConnection()的影响是什么?

3)整个管道概念如何运作?它是否像动态Lua?将此Java代码转换为Lua脚本并将其作为脚本发送到Redis,在Redis中执行并返回?在这次回调中感到惊讶;代码也在访问/更新外部类变量,那么所有变量会发生什么?所有那些外部类变量也发送到lua中的redis?

4)我看到许多关于doInRedis API返回null的示例;为什么这样?如何从中返回/获取有效的对象?

1 个答案:

答案 0 :(得分:3)

您的大部分问题都可以在Spring Data Redis reference documentation中找到。

在深入研究Pipelining之前,一个Hash中的单个多次获取不需要Pipelining,因为它只是一个命令。流水线技术不会提高Redis交互的性能/稳定性。

Pipelining被安排为回调,用于在不等待结果的情况下发出多个命令 - 将其视为一个批处理,您可以在以后获得所有结果。由于流水线操作会在最后同步响应,因此您不会在回调中接收结果值,但最后会在流水线会话同步且executePipelined(…)终止时收到。

您的代码应该看起来像:

List<Object> results = getRedisTemplate().executePipelined(new RedisCallback<Object>() {

    @Override
    public Object doInRedis(RedisConnection connection) {

            connection.hMGet(key.getBytes(), params);

            return null;
    }
});

List<Object> hmget = (List<Object>) results.get(0);

您必须仅使用您作为回调参数接收的连接,因为连接已进入流水线模式。从回调外部获取连接(如template.getConnectionFactory().getConnection())将打开一个新连接并执行Redis命令并等待响应 - 没有流水线应用于任何外部获得的连接。

您也可以使用RedisTemplate的方法,而不是使用普通连接。 executePipelined(…)将回调中使用的连接绑定到当前Thread,如果调用模板API方法,则重用该绑定连接。

关于你的Lua问题:代码/方法调用不会转换为Lua。