redis 五種數據結構詳解(string,list,set,zset,hash)

redis 五種數據結構詳解(string,list,set,zset,hash)

Redis不只僅支持簡單的key-value類型的數據,同時還提供list,set,zset,hash等數據結構的存儲。php

String——字符串
Hash——字典
List——列表
Set——集合
Sorted Set——有序集合redis

下面咱們就來簡單說明一下它們各自的使用場景:數據庫

1、Redis String類型

string類型的數據存儲是最簡單的key-value存儲;數組

1.string字符串讀寫實現方式:

    public function testRedis(){
        //string類型的數據結構
        app()->redis->set('1', 'aa');
    //根據key取出value值
    $string = app()->redis->get('1');
    }

2.redis客戶端查看結果:

3.string字符串的其餘redis操做方法:

//普通set/get操做
$redis->set('library', 'predis');
$retval = $redis->get('library');
echo $retval; //顯示 'predis'數據結構

//setex set一個存儲時效
$redis->setex('str', 10, 'bar'); //表示存儲有效期爲10秒app

//setnx/msetnx至關於add操做,不會覆蓋已有值
$redis->setnx('foo',12); //true
$redis->setnx('foo',34); //falsedom

//getset操做,set的變種,結果返回替換前的值
$redis->getset('foo',56);//返回34網站

// incrby/incr/decrby/decr 對值的遞增和遞減
$redis->incr('foo'); //foo爲57
$redis->incrby('foo',2); //foo爲59spa

//exists檢測是否存在某值
$redis->exists('foo');//true線程

//del 刪除
$redis->del('foo');//true

//type 類型檢測,字符串返回string,列表返回 list,set表返回set/zset,hash表返回hash
$redis->type('foo');//不存在,返回none
$redis->set('str','test');
$redis->type('str'); //字符串,返回string

//append 鏈接到已存在字符串
$redis->append('str','_123'); //返回累加後的字符串長度8,此進str爲 'test_123'

//setrange 部分替換操做
$redis->setrange('str',0,'abc'); //返回3,參數2爲0時等同於set操做
$redis->setrange('str',2,'cd');//返回4,表示從第2個字符後替換,這時'str'爲'abcd'

//substr 部分獲取操做
$redis->substr('str',0,2);//表示從第0個起,取到第2個字符,共3個,返回'abc'

//strlen 獲取字符串長度
$redis->strlen('str'); //返回4

//setbit/getbit 位存儲和獲取
$redis->setbit('binary',31,1);  //表示在第31位存入1,這邊可能會有大小端問題?不過不要緊,getbit 應該不會有問題
$redis->getbit('binary',31);    //返回1

//keys 模糊查找功能,支持*號以及?號(匹配一個字符)
$redis->set('foo1',123);
$redis->set('foo2',456);
$redis->keys('foo*'); //返回foo1和foo2的array
$redis->keys('f?o?');  //同上

//randomkey 隨機返回一個key
$redis->randomkey(); //多是返回 'foo1'或者是'foo2'及其它任何一存在redis的key

//rename/renamenx 對key進行更名,所不一樣的是renamenx不容許改爲已存在的key
$redis->rename('str','str2'); //把原先命名爲'str'的key改爲了'str2'

//expire 設置key-value的時效性,ttl 獲取剩餘有效期,persist 從新設置爲永久存儲
$redis->expire('foo', 1); //設置有效期爲1秒
$redis->ttl('foo'); //返回有效期值1s
$redis->expire('foo'); //取消expire行爲

//dbsize 返回redis當前數據庫的記錄總數
$redis->dbsize();

2、Redis Hash表

 redis中hash表存儲數據,比較相似數據庫中表的一條記錄;

1.hash讀寫實現方式:

    public function testRedis()
    {//存儲 hash類型 數據結構
        app()->redis->hset('goods', 'apple', '蘋果');

        //取出 hash表中的數據
        $hash = app()->redis->hget('goods', 'apple');
        print_r($hash);
        echo "\n";
        
        die();
    }

2.redis客戶端查看結果:

 

 3.關於hash表其餘redis操做方法:

//hset/hget 存取hash表的數據
$redis->hset('hash1','key1','v1'); //將key爲'key1' value爲'v1'的元素存入hash1表
$redis->hset('hash1','key2','v2');
$redis->hget('hash1','key1');  //取出表'hash1'中的key 'key1'的值,返回'v1'

//hexists 返回hash表中的指定key是否存在
$redis->hexists ('hash1','key1'); //true or false

//hdel 刪除hash表中指定key的元素
$redis->hdel('hash1','key2'); //true or false

//hlen 返回hash表元素個數
$redis->hlen('hash1'); //1

//hsetnx 增長一個元素,但不能重複
$redis->hsetnx('hash1','key1','v2'); //false
$redis->hsetnx('hash1','key2','v2'); //true

//hmset/hmget 存取多個元素到hash表
$redis->hmset('hash1',array('key3'=>'v3','key4'=>'v4'));
$redis->hmget('hash1',array('key3','key4')); //返回相應的值 array('v3','v4')

//hincrby 對指定key進行累加
$redis->hincrby('hash1','key5',3); //返回3
$redis->hincrby('hash1','key5',10); //返回13

//hkeys 返回hash表中的全部key
$redis->hkeys('hash1'); //返回array('key1','key2','key3','key4','key5')

//hvals 返回hash表中的全部value
$redis->hvals('hash1'); //返回array('v1','v2','v3','v4',13)

//hgetall 返回整個hash表元素
$redis->hgetall('hash1'); //返回array('key1'=>'v1','key2'=>'v2','key3'=>'v3','key4'=>'v4','key5'=>13)

3、Redis list列表

List數據結構是鏈表結構,是雙向的,能夠在鏈表左,右兩邊分別操做;

也能夠把list當作一種隊列,因此在不少時候能夠用redis用做消息隊列,這個時候它的做用就相似於activeMq啦;

應用案例有時間軸數據,評論列表,消息傳遞等等,它能夠提供簡便的分頁,讀寫操做。

1.list 讀寫實現方式:

    public function testRedis()
    {//存儲 list
        app()->redis->lpush('news', 'cc'); //從隊列前面插入元素
        app()->redis->lpush('news', 'ee'); //從隊列前面插入元素
        app()->redis->rpush('news', 'dd');//從隊列後面插入元素
        $list = app()->redis->lrange('news', 0, -1);//取出list全部元素
        print_r($list);
        echo "\n";

        die();
    }

2.在redis客戶端中直觀查看list的存儲結果:

3.list列表的其餘redis操做方法:

//rpush/rpushx 有序列表操做,從隊列後插入元素
//lpush/lpushx 和rpush/rpushx的區別是插入到隊列的頭部,同上,'x'含義是隻對已存在的key進行操做
$redis->rpush('fooList', 'bar1'); //返回一個列表的長度1
$redis->lpush('fooList', 'bar0'); //返回一個列表的長度2
$redis->rpushx('fooList', 'bar2'); //返回3,rpushx只對已存在的隊列作添加,不然返回0
//llen返回當前列表長度
$redis->llen('fooList');//3

//lrange 返回隊列中一個區間的元素
$redis->lrange('fooList',0,1); //返回數組包含第0個至第1個共2個元素
$redis->lrange('fooList',0,-1);//返回第0個至倒數第一個,至關於返回全部元素,注意redis中不少時候會用到負數,下同

//lindex 返回指定順序位置的list元素
$redis->lindex('fooList',1); //返回'bar1'

//lset 修改隊列中指定位置的value
$redis->lset('fooList',1,'123');//修改位置1的元素,返回true

//lrem 刪除隊列中左起指定數量的字符
$redis->lrem('fooList',1,'_'); //刪除隊列中左起(右起使用-1)1個字符'_'(如有)

//lpop/rpop 相似棧結構地彈出(並刪除)最左或最右的一個元素
$redis->lpop('fooList'); //'bar0'
$redis->rpop('fooList'); //'bar2'

//ltrim 隊列修改,保留左邊起若干元素,其他刪除
$redis->ltrim('fooList', 0,1); //保留左邊起第0個至第1個元素

//rpoplpush 從一個隊列中pop出元素並push到另外一個隊列
$redis->rpush('list1','ab0');
$redis->rpush('list1','ab1');
$redis->rpush('list2','ab2');
$redis->rpush('list2','ab3');
$redis->rpoplpush('list1','list2');//結果list1 =>array('ab0'),list2 =>array('ab1','ab2','ab3')
$redis->rpoplpush('list2','list2');//也適用於同一個隊列,把最後一個元素移到頭部list2 =>array('ab3','ab1','ab2')

//linsert 在隊列的中間指定元素前或後插入元素
$redis->linsert('list2', 'before','ab1','123'); //表示在元素'ab1'以前插入'123'
$redis->linsert('list2', 'after','ab1','456');   //表示在元素'ab1'以後插入'456'

//blpop/brpop 阻塞並等待一個列隊不爲空時,再pop出最左或最右的一個元素(這個功能在php之外能夠說很是好用)
//brpoplpush 一樣是阻塞並等待操做,結果同rpoplpush同樣
$redis->blpop('list3',10); //若是list3爲空則一直等待,直到不爲空時將第一元素彈出,10秒後超時;

4、Redis Set集合

Set 就是一個集合,集合的概念就是一堆不重複值的組合。利用 Redis 提供的 Set 數據結構,能夠存儲一些集合性的數據。

好比在微博應用中,能夠將一個用戶全部的關注人存在一個集合中,將其全部粉絲存在一個集合。

由於 Redis 很是人性化的爲集合提供了求交集、並集、差集等操做,那麼就能夠很是方便的實現如共同關注、共同喜愛、二度好友等功能,對上面的全部集合操做,你還可使用不一樣的命令選擇將結果返回給客戶端仍是存集到一個新的集合中。

1.共同好友、二度好友
2.利用惟一性,能夠統計訪問網站的全部獨立 IP
3.好友推薦的時候,根據 tag 求交集,大於某個 threshold 就能夠推薦

1.set 讀寫實現方式:

    public function testRedis()
    {//存儲 set
        $fans = app()->redis->sadd('fans', 'ff');
        if($fans){
            print_r('set add ff success');
        }else{
            print_r('set add ff fail');
        }
        $fans = app()->redis->sadd('fans', 'gg'); //不存在返回true
        if($fans){
            print_r('set add gg success');
        }else{
            print_r('set add gg fail');
        }
        $fans = app()->redis->sadd('fans', 'gg'); //不存在返回false
        if($fans){
            print_r('set add gg success');
        }else{
            print_r('set add gg fail');
        }
        //取出set
        $fans = app()->redis->smembers('fans');
        print_r($fans);
        echo "\n";
    }

2.在redis客戶端中直觀查看set集合的存儲結果:

3.set集合的其餘redis操做方法:

//sadd 增長元素,返回true,重複返回false
$redis->sadd('set1','ab');
$redis->sadd('set1','cd');
$redis->sadd('set1','ef');

//srem 移除指定元素
$redis->srem('set1','cd'); //刪除'cd'元素

//spop 彈出首元素
$redis->spop('set1');

//smove 移動當前set表的指定元素到另外一個set表
$redis->sadd('set2','123');
$redis->smove('set1','set2','ab');//移動'set1'中的'ab'到'set2',返回true or false

//scard 返回當前set表元素個數
$redis->scard('set2');//2

//sismember 判斷元素是否屬於當前表
$redis->sismember('set2','123'); //true or false

//smembers 返回當前表的全部元素
$redis->smembers('set2'); //array('123','ab');

//sinter/sunion/sdiff  返回兩個表中元素的交集/並集/補集
$redis->sadd('set1','ab');
$redis->sinter('set2','set1'); //返回array('ab')

//sinterstore/sunionstore/sdiffstore 將兩個表交集/並集/補集元素copy到第三個表中
$redis->set('foo',0);
$redis->sinterstore('foo','set1'); //這邊等同於將'set1'的內容copy到'foo'中,並將'foo'轉爲set表
$redis->sinterstore('foo',array('set1','set2')); //將'set1'和'set2'中相同的元素copy到'foo'表中,覆蓋'foo'原有內容

//srandmember 返回表中一個隨機元素
$redis->srandmember('set1');

5、Redis Zset集合(Sorted Sets)

zset是set的一個升級版本,他在set的基礎上增長了一個順序屬性,這一屬性在添加修改元素的時候能夠指定,每次指定後,zset會自動從新按新的值調整順序。 能夠對指定鍵的值進行排序權重的設定,它應用排名模塊比較多。

好比一個存儲全班同窗成績的 Sorted Sets,其集合 value 能夠是同窗的學號,而 score 就能夠是其考試得分,這樣在數據插入集合的時候,就已經進行了自然的排序。另外還能夠用 Sorted Sets 來作帶權重的隊列,好比普通消息的 score 爲1,重要消息的 score 爲2,而後工做線程能夠選擇按 score 的倒序來獲取工做任務,讓重要的任務優先執行。

zset集合能夠完成有序執行、按照優先級執行的狀況;

1.zset 讀寫實現方式:

    public function testRedis()
    {//zset 添加元素
        app()->redis->zadd('students', '1', '90');
        app()->redis->zadd('students', '2', '80');
        app()->redis->zadd('students', '3', '95');
        app()->redis->zadd('students', '7', '75');
        app()->redis->zadd('students', '5', '55');
        //取出 zset
        $zset = app()->redis->zrange('students', 0, -1);
        print_r($zset);
        echo "\n";
    }

2.在redis客戶端中直觀查看zset集合的存儲結果:

3.zset集合的其餘redis操做方法:

//sadd 增長元素,並設置序號,返回true,重複返回false
$redis->zadd('zset1',1,'ab');
$redis->zadd('zset1',2,'cd');
$redis->zadd('zset1',3,'ef');

//zincrby 對指定元素索引值的增減,改變元素排列次序
$redis->zincrby('zset1',10,'ab');//返回11

//zrem 移除指定元素
$redis->zrem('zset1','ef'); //true or false

//zrange 按位置次序返回表中指定區間的元素
$redis->zrange('zset1',0,1); //返回位置0和1之間(兩個)的元素
$redis->zrange('zset1',0,-1);//返回位置0和倒數第一個元素之間的元素(至關於全部元素)

//zrevrange 同上,返回表中指定區間的元素,按次序倒排
$redis->zrevrange('zset1',0,-1); //元素順序和zrange相反

//zrangebyscore/zrevrangebyscore 按順序/降序返回表中指定索引區間的元素
$redis->zadd('zset1',3,'ef');
$redis->zadd('zset1',5,'gh');
$redis->zrangebyscore('zset1',2,9); //返回索引值2-9之間的元素 array('ef','gh')
//參數形式
$redis->zrangebyscore('zset1',2,9,'withscores'); //返回索引值2-9之間的元素幷包含索引值 array(array('ef',3),array('gh',5))
$redis->zrangebyscore('zset1',2,9,array('withscores' =>true,'limit'=>array(1, 2))); //返回索引值2-9之間的元素,'withscores' =>true表示包含索引值; 'limit'=>array(1, 2),表示最多返回2條,結果爲array(array('ef',3),array('gh',5))

//zunionstore/zinterstore 將多個表的並集/交集存入另外一個表中
$redis->zunionstore('zset3',array('zset1','zset2','zset0')); //將'zset1','zset2','zset0'的並集存入'zset3'
//其它參數
$redis->zunionstore('zset3',array('zset1','zset2'),array('weights' => array(5,0)));//weights參數表示權重,其中表示並集後值大於5的元素排在前,大於0的排在後
$redis->zunionstore('zset3',array('zset1','zset2'),array('aggregate' => 'max'));//'aggregate' => 'max'或'min'表示並集後相同的元素是取大值或是取小值

//zcount 統計一個索引區間的元素個數
$redis->zcount('zset1',3,5);//2
$redis->zcount('zset1','(3',5)); //'(3'表示索引值在3-5之間但不含3,同理也可使用'(5'表示上限爲5但不含5

//zcard 統計元素個數
$redis->zcard('zset1');//4

//zscore 查詢元素的索引
$redis->zscore('zset1','ef');//3

//zremrangebyscore 刪除一個索引區間的元素
$redis->zremrangebyscore('zset1',0,2); //刪除索引在0-2之間的元素('ab','cd'),返回刪除元素個數2

//zrank/zrevrank 返回元素所在表順序/降序的位置(不是索引)
$redis->zrank('zset1','ef');//返回0,由於它是第一個元素;zrevrank則返回1(最後一個)

//zremrangebyrank 刪除表中指定位置區間的元素$redis->zremrangebyrank('zset1',0,10); //刪除位置爲0-10的元素,返回刪除的元素個數2

相關文章
相關標籤/搜索