使用Python操做Redis

1. 安裝pyredispython

首先安裝pipredis


<SHELL># apt-get install python-pip數據庫

......安全

<SHELL># pip install --proxy=http://172.1.2.6:8080 redis服務器

  Downloading redis-2.9.1.tar.gz (62kB): 62kB downloadedapp

  Running setup.py (path:/tmp/pip_build_root/redis/setup.py) egg_info for package redisdom

  ......ide

  Successfully installed redis網站

  Cleaning up...ui

也可使用easy_install的方式來安裝:



easy_install redis

或者直接編譯安裝:




wget https://pypi.python.org/packages/source/r/redis/redis-2.9.1.tar.gz

tar xvzf redis-2.9.1.tar.gz

cd redis-2.9.1

python setup.py install


2 . 簡單的redis操做

redis鏈接實例是線程安全的,能夠直接將redis鏈接實例設置爲一個全局變量,直接使用。若是須要另外一個Redis實例(or Redis數據庫)時,就須要從新建立redis鏈接實例來獲取一個新的鏈接。同理,python的redis沒有實現select命令。



>>> import redis


>>> r = redis.Redis(host='localhost',port=6379,db=0)


>>> r.set('guo','shuai')


True


>>> r.get('guo')


'shuai'

>>> r['guo']            

'shuai'

>>> r.keys()

['guo']

>>> r.dbsize()         #當前數據庫包含多少條數據      

1L

>>> r.delete('guo')

1

>>> r.save()               #執行「檢查點」操做,將數據寫回磁盤。保存時阻塞

True

>>> r.get('guo');

>>> r.flushdb()        #清空r中的全部數據

True



3. pipeline操做

管道(pipeline)是redis在提供單個請求中緩衝多條服務器命令的基類的子類。它經過減小服務器-客戶端之間反覆的TCP數據庫包,從而大大提升了執行批量命令的功能。



>>> p = r.pipeline()        --建立一個管道


>>> p.set('hello','redis')


>>> p.sadd('faz','baz')


>>> p.incr('num')


>>> p.execute()


[True, 1, 1]

>>> r.get('hello')

'redis'

管道的命令能夠寫在一塊兒,如:



1

>>> p.set('hello','redis').p.sadd('faz','baz').incr('num').execute()


默認的狀況下,管道里執行的命令能夠保證執行的原子性,


執行pipe = r.pipeline(transaction=False)能夠禁用這一特性。


4. 應用場景 – 頁面點擊數

《Redis Cookbook》對這個經典場景進行詳細描述。假定咱們對一系列頁面須要記錄點擊次數。例如論壇的每一個帖子都要記錄點擊次數,而點擊次數比回帖的次數的多得多。若是使用關係數據庫來存儲點擊,可能存在大量的行級鎖爭用。因此,點擊數的增長使用redis的INCR命令最好不過了。

當redis服務器啓動時,能夠從關係數據庫讀入點擊數的初始值(1237這個頁面被訪問了34634次)



>>> r.set("visit:1237:totals",34634)

True

每當有一個頁面點擊,則使用INCR增長點擊數便可。



>>> r.incr("visit:1237:totals")

34635

>>> r.incr("visit:1237:totals")

34636

頁面載入的時候則可直接獲取這個值




>>> r.get ("visit:1237:totals")

'34636'



5. 使用hash類型保存多樣化對象

當有大量類型文檔的對象,文檔的內容都不同時,(即「表」沒有固定的列),可使用hash來表達。




>>> r.hset('users:jdoe',  'name', "John Doe")

1L

>>> r.hset('users:jdoe', 'email', 'John@test.com')

1L

>>> r.hset('users:jdoe',  'phone', '1555313940')

1L

>>> r.hincrby('users:jdoe', 'visits', 1)

1L

>>> r.hgetall('users:jdoe')

{'phone': '1555313940', 'name': 'John Doe', 'visits': '1', 'email': 'John@test.com'}

>>> r.hkeys('users:jdoe')

['name', 'email', 'phone', 'visits']



6. 應用場景 – 社交圈子數據

在社交網站中,每個圈子(circle)都有本身的用戶羣。經過圈子能夠找到有共同特徵(好比某一體育活動、遊戲、電影等愛好者)的人。當一個用戶加入一個或幾個圈子後,系統能夠向這個用戶推薦圈子中的人。

咱們定義這樣兩個圈子,並加入一些圈子成員。



>>> r.sadd('circle:game:lol','user:debugo')

1

>>> r.sadd('circle:game:lol','user:leo')

1

>>> r.sadd('circle:game:lol','user:Guo')

1

>>> r.sadd('circle:soccer:InterMilan','user:Guo')

1

>>> r.sadd('circle:soccer:InterMilan','user:Levis')

1

>>> r.sadd('circle:soccer:InterMilan','user:leo')

1

#得到某一圈子的成員




>>> r.smembers('circle:game:lol')

set(['user:Guo', 'user:debugo', 'user:leo'])

redis> smembers circle:jdoe:family    

可使用集合運算來獲得幾個圈子的共同成員:




>>> r.sinter('circle:game:lol', 'circle:soccer:InterMilan')

set(['user:Guo', 'user:leo'])


>>> r.sunion('circle:game:lol', 'circle:soccer:InterMilan')

set(['user:Levis', 'user:Guo', 'user:debugo', 'user:leo'])



7. 應用場景 – 實時用戶統計

Counting Online Users with Redis介紹了這個方法。當咱們須要在頁面上顯示當前的在線用戶時,就可使用Redis來完成了。首先得到當前時間(以Unix timestamps方式)除以60,能夠基於這個值建立一個key。而後添加用戶到這個集合中。當超過你設定的最大的超時時間,則將這個集合設爲過時;而當須要查詢當前在線用戶的時候,則將最後N分鐘的集合交集在一塊兒便可。因爲redis鏈接對象是線程安全的,因此能夠直接使用一個全局變量來表示。


import time

from redis import Redis

from datetime import datetime

ONLINE_LAST_MINUTES = 5

redis = Redis()

 

def mark_online(user_id):         #將一個用戶標記爲online

    now = int(time.time())        #當前的UNIX時間戳

    expires = now + (app.config['ONLINE_LAST_MINUTES'] * 60) + 10    #過時的UNIX時間戳

    all_users_key = 'online-users/%d' % (now // 60)        #集合名,包含分鐘信息

    user_key = 'user-activity/%s' % user_id                

    p = redis.pipeline()

    p.sadd(all_users_key, user_id)                         #將用戶id插入到包含分鐘信息的集合中

    p.set(user_key, now)                                   #記錄用戶的標記時間

    p.expireat(all_users_key, expires)                     #設定集合的過時時間爲UNIX的時間戳

    p.expireat(user_key, expires)

    p.execute()

 

def get_user_last_activity(user_id):        #得到用戶的最後活躍時間

    last_active = redis.get('user-activity/%s' % user_id)  #若是獲取不到,則返回None

    if last_active is None:

        return None

    return datetime.utcfromtimestamp(int(last_active))

 

def get_online_users():                     #得到當前online用戶的列表

    current = int(time.time()) // 60        

    minutes = xrange(app.config['ONLINE_LAST_MINUTES'])

    return redis.sunion(['online-users/%d' % (current - x)        #取ONLINE_LAST_MINUTES分鐘對應集合的交集

                         for x in minutes])




import redis

 

r = redis.Redis(host=’localhost’, port=6379, db=0)

 

r['test'] = ‘test’ #或者能夠r.set(‘test’, ‘test’) 設置key

 

r.get(‘test’)  #獲取test的值

 

r.delete(‘test’) #刪除這個key

 

r.flushdb() #清空數據庫

 

r.keys() #列出全部key

 

r.exists(‘test’) #檢測這個key是否存在

 

r.dbsize() #數據庫中多少個條數



  


info = r.info() 
for key in info: 
  print "%s: %s" % (key, info[key])

# 查數據庫大小 
print '\ndbsize: %s' % r.dbsize()

# 看鏈接 
print "ping %s" % r.ping()

# 選數據庫 
#r.select(2)

# 移動數據去2數據庫 
#r.move('a',2)

# 其餘 
#r.save('a') # 存數據 
#r.lastsave('a') # 取最後一次save時間 
#r.flush()  #刷新 
#r.shutdown() #關閉全部客戶端,停掉全部服務,退出服務器 

#-------------------------------------------- 
# 它有四種類型: string(key,value)、list(序列)、set(集合)、zset(有序集合,多了一個順序屬性) 
# 不知道你用的哪一種類型? 
# print r.get_type('a') #能夠告訴你 
# ------------------------------------------- 
# string操做 
print '-'*20 
# 塞數據 
r['c1'] = 'bar' 
#或者 
r.set('c2','bar') 
#這裏有個 getset屬性,若是爲True 能夠在存新數據時將上次存儲內容同時搞出來 
print 'getset:',r.getset('c2','jj') 
#若是你想設置一個遞增的整數 每執行一次它自加1: 
print 'incr:',r.incr('a') 
#若是你想設置一個遞減的整數 please: 
print 'decr:',r.decr('a')

# 取數據 
print 'r['']:',r['c1'] 
#或者 
print 'get:',r.get('a') 
#或者 同時取一批 
print 'mget:',r.mget('c1','c2') 
#或者 同時取一批 它們的名字(key)很像 而剛好你又不想輸所有 
print 'keys:',r.keys('c*') 
#又或者 你只想隨機取一個: 
print 'randomkey:',r.randomkey()


# 查看一個數據有沒有 有 1 無0 
print 'existes:',r.exists('a')

# 刪數據 1是刪除成功 0和None是沒這個東西 
print 'delete:',r.delete('cc') 
# 哦對了 它是支持批量操做的 
print 'delete:',r.delete('c1','c2')

# 其餘 
r.rename('a','c3') #呃.更名 
r.expire('c3',10) #讓數據10秒後過時 說實話我不太明白麼意思 
r.ttl('c3') #看剩餘過時時間 不存在返回-1

#-------------------------------- 
# 序列(list)操做 
print '-'*20 
# 它是兩頭通的 
# 塞入 
r.push('b','gg') 
r.push('b','hh') 
# head 屬性控制是否是從另外一頭塞 
r.push('b','ee',head=True) 
# 看長度 
print 'list len:',r.llen('b') 
# 列出一批出來 
print 'list lrange:',r.lrange('b',start=0,end=-1)

# 取出一位 
print 'list index 0:',r.lindex('b',0)

# 修剪列表 
#若start 大於end,則將這個list清空 
print 'list ltrim :',r.ltrim('b',start=0,end=3) #只留 從0到3四位

# 排序 
# 這但是個大工程

#-------------------------------- 
# 集合(set)操做 
# 塞數據 
r.sadd('s', 'a')

# 判斷一個set長度爲多少 不存在爲0 
r.scard('s')

# 判斷set中一個對象是否存在 
r.sismember('s','a')

# 求交集 
r.sadd('s2','a') 
r.sinter('s1','s2')

#求交集並將結果賦值 
r.sinterstore('s3','s1','s2')

# 看一個set對象 
r.smembers('s3')

# 求並集 
r.sunion('s1','s2') 

#求並集 並將結果返回 
r.sunionstore('ss','s1','s2','s3')

# 求不一樣 
# 在s1中有,但在s2和s3中都沒有的數 
r.sdiff('s1','s2','s3') 
r.sdiffstore('s4','s1','s2')# 這個你懂的

# 取個隨機數 
r.srandmember('s1')

#------------------------------------- #zset 有序set #'zadd', 'zcard', 'zincr', 'zrange', 'zrangebyscore', 'zrem', 'zscore' # 分別對應 #添加, 數量, 自加1,取數據,按照積分(範圍)取數據,刪除,取積分

相關文章
相關標籤/搜索