天天看點

redis 五種資料結構詳解(string,list,set,zset,hash)redis 五種資料結構詳解(string,list,set,zset,hash)

redis 五種資料結構詳解(string,list,set,zset,hash)

Redis不僅僅支援簡單的key-value類型的資料,同時還提供list,set,zset,hash等資料結構的存儲。

String——字元串

Hash——字典

List——清單

Set——集合

Sorted Set——有序集合

下面我們就來簡單說明一下它們各自的使用場景:

一、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用戶端檢視結果:

redis 五種資料結構詳解(string,list,set,zset,hash)redis 五種資料結構詳解(string,list,set,zset,hash)

3.string字元串的其他redis操作方法:

//普通set/get操作

$redis->set('library', 'predis');

$retval = $redis->get('library');

echo $retval; //顯示 'predis'

//setex set一個存儲時效

$redis->setex('str', 10, 'bar'); //表示存儲有效期為10秒

//setnx/msetnx相當于add操作,不會覆寫已有值

$redis->setnx('foo',12); //true

$redis->setnx('foo',34); //false

//getset操作,set的變種,結果傳回替換前的值

$redis->getset('foo',56);//傳回34

// incrby/incr/decrby/decr 對值的遞增和遞減

$redis->incr('foo'); //foo為57

$redis->incrby('foo',2); //foo為59

//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();

二、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();
    }      

redis 五種資料結構詳解(string,list,set,zset,hash)redis 五種資料結構詳解(string,list,set,zset,hash)

 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)

三、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的存儲結果:

redis 五種資料結構詳解(string,list,set,zset,hash)redis 五種資料結構詳解(string,list,set,zset,hash)

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秒後逾時;

四、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集合的存儲結果:

redis 五種資料結構詳解(string,list,set,zset,hash)redis 五種資料結構詳解(string,list,set,zset,hash)

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->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');

五、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集合的存儲結果:

redis 五種資料結構詳解(string,list,set,zset,hash)redis 五種資料結構詳解(string,list,set,zset,hash)

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',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

原文位址cnblogs.com