我想在Redis中维护一些元数据。
meta_key = build_key() meta_data = { "user": 12345, "tag": "D12321341234123", } res = redis_sip.hmset(meta_key, meta_data)
它按预期工作。
现在,我想在此meta_data结构中保留一个列表,并能够将元素添加到列表中。
例如:
meta_data = { "user": 12345, "tag": "D12321341234123", "items": [] }
但这立即引发异常:
redis.exceptions.DataError: Invalid input of type: 'list'. Convert to a byte, string or number first.
我认为我可以创建一个新密钥并用于zadd维护列表。但是我想尽量减少密钥的数量。这是因为用户注销后,我需要快速使密钥失效。将密钥保持在最低限度可以帮助我
zadd
1)快速移出按键
2)避免出现错误,因为用于保持制表符的键较少
有什么办法可以使列表保持redis值并轻松扩展列表?
在大多数情况下,SADD或ZADD使用流水线命令会更好。如果存在另一个客户端可能会在其间获取密钥的风险,请使用MULTI / EXEC事务,从而获得不完整的对象。
SADD
ZADD
在某些情况下,在哈希字段中对列表进行字符串化可能是合理的。
关于“快速退出键”,请确保使用UNLINK代替DEL。
UNLINK
DEL
如果选择进行字符串化,则以下是使用Lua和Lua CJSON库在散列字段中自动支持插入和删除到JSON编码数组的方法:
插入 :
local items = cjson.decode(redis.call('HGET', KEYS[1], 'items')) table.insert(items, ARGV[1]) return redis.call('HSET', KEYS[1], 'items', cjson.encode(items))
按值删除 :
local items = cjson.decode(redis.call('HGET', KEYS[1], 'items')) local pos = -1; for i, v in ipairs(items) do if ARGV[1] == v then pos = i break end end if pos == -1 then return -1 else table.remove(items, pos) return redis.call('HSET', KEYS[1], 'items', cjson.encode(items)) end
用法示例 :
> HGETALL meta_key 1) "user" 2) "12345" 3) "tag" 4) "D12321341234123" 5) "items" 6) "{}" > EVAL "local items = cjson.decode(redis.call('HGET', KEYS[1], 'items')) \n table.insert(items, ARGV[1]) \n return redis.call('HSET', KEYS[1], 'items', cjson.encode(items))" 1 meta_key value1 (integer) 0 > HGETALL meta_key 1) "user" 2) "12345" 3) "tag" 4) "D12321341234123" 5) "items" 6) "[\"value1\"]" > EVAL "local items = cjson.decode(redis.call('HGET', KEYS[1], 'items')) \n table.insert(items, ARGV[1]) \n return redis.call('HSET', KEYS[1], 'items', cjson.encode(items))" 1 meta_key value2 (integer) 0 > HGETALL meta_key 1) "user" 2) "12345" 3) "tag" 4) "D12321341234123" 5) "items" 6) "[\"value1\",\"value2\"]" > EVAL "local items = cjson.decode(redis.call('HGET', KEYS[1], 'items')) \n local pos = -1; \n for i, v in ipairs(items) do \n if ARGV[1] == v then \n pos = i \n break \n end \n end \n if pos == -1 then \n return -1 \n else \n table.remove(items, pos) \n return redis.call('HSET', KEYS[1], 'items', cjson.encode(items)) \n end" 1 meta_key value1 (integer) 0 > HGETALL meta_key 1) "user" 2) "12345" 3) "tag" 4) "D12321341234123" 5) "items" 6) "[\"value2\"]" > EVAL "local items = cjson.decode(redis.call('HGET', KEYS[1], 'items')) \n local pos = -1; \n for i, v in ipairs(items) do \n if ARGV[1] == v then \n pos = i \n break \n end \n end \n if pos == -1 then \n return -1 \n else \n table.remove(items, pos) \n return redis.call('HSET', KEYS[1], 'items', cjson.encode(items)) \n end" 1 meta_key value3 (integer) -1 > HGETALL meta_key 1) "user" 2) "12345" 3) "tag" 4) "D12321341234123" 5) "items" 6) "[\"value2\"]"