我对redis相当新鲜。我有一个CMS,在这样的结构中将JSON输出到redis:
partial:1 => {id: 1, title: 'Foo1', ...}
partial:2 => {id: 2, title: 'Foo2', ...}
partial:3 => {id: 3, title: 'Foo3', ...}
page:home => {
id: 'Homepage',
teaser1: 'partial:1',
teaser2: 'partial:2',
teaser3: {type: Slider, content: 'partial:3'}
}
因此JSON可以包含符合其结构中某个命名方案的其他redis-key。我想要的是一种查询redis的方法,这样当我得到page:home
- 键时,对json中其他键的引用得到了扩展'各自的价值观,如下:
{
id: 'Homepage',
teaser1: {id: 1, title: 'Foo1', ...},
teaser2: {id: 2, title: 'Foo2', ...},
teaser3: {type: Slider, content: {id: 3, title: 'Foo3', ...}
}
这可能吗?如何实现?
答案 0 :(得分:2)
是。这是很有可能。这是一种有效的方法:
创建一个函数将redis哈希值(例如:partial:*,page:home)转换为lua表: hgetall
创建一个函数来检查给定的redis hashname是否符合您的命名方案,如果是,请使用 hgetall 将其转换为lua表;否则返回相同的值: evaluate_hash
这是一个简单的实现:
- 脚本:redis_expand_as_json.lua
--Function to Convert a given hash name (e.g: partial:1..n, page:home) to a lua table
local function hgetall(a)local b=redis.call('HGETALL',a)local c={}local d;for e,f in ipairs(b)do if e%2==1 then d=f else c[d]=f end end;return c end
--Function to check if the given value conforms with a naming scheme,
-- if so convert it to a lua table; otherwise return the values as is.
local function evaluate_hash(value)
local pattern = "partial:%d+"
if string.match(value, pattern) then
return hgetall(value)
else
return value
end
end
--Function to convert a given hash_name to a lua table,
-- iterate all elements and convert them to lua table if necessary
-- returns the table as a json object
local function expand(hash_name)
local obj_table = hgetall(hash_name)
for k, val in pairs(obj_table) do
obj_table[k] = evaluate_hash(val)
end
return cjson.encode(obj_table)
end
local page = KEYS[1]
local json_result = expand(page) or {}
redis.log(redis.LOG_NOTICE, tostring(json_result))
return json_result
在控制台中测试:
redis-cli -c hmset“partial:1”id 1 title foo1
行
redis-cli -c hmset“partial:2”id 2 title foo2
行
redis-cli -c hmset'page:home'id'主页'teaser1'部分:1'teaser2'partial:2'
行
redis-cli EVAL“$(cat redis_expand_as_json.lua)”1'页:主页'
{"teaser1":{"id":"1","title":"foo1"},"teaser2":{"id":"2","title":"foo2"},"id":"Homepage"}
答案 1 :(得分:0)
我不认识Redis,但也许这适合你:
local T=[[
partial:1 => {id: 1, title: 'Foo1', ...}
partial:2 => {id: 2, title: 'Foo2', ...}
partial:3 => {id: 3, title: 'Foo3', ...}
page:home => {
id: 'Homepage',
teaser1: 'partial:1',
teaser2: 'partial:2',
teaser3: {type: Slider, content: 'partial:3'}
}
]]
local D={}
for k,v in T:gmatch("(%w+:%w+)%s*=>%s*(%b{})") do
D[k]=v
end
print((T:gsub("'(.-)'",D)))