什么是redis:key-value存储系统

计算机的处理速度:cpu > 内存 > 硬盘,而mysql是以文件形式存在硬盘的。读写太频繁会撑不住。

第一种、页面缓存 ==>>适合静态界面,比如新闻界面,CMS系统很常见。

第二种、数据缓存==>>适合一些数据会发生变动的界面。比如京东搜索商品得到的界面,数据界面常见。

 

redis:高速缓存数据库,属于数据缓存。具有可持久化。采用C语言开发==>>可持久化指的是,一边运行,一边写入到硬盘。所以还是很安全的。

 

redis对比memcache:==>>

1、redis提供的数据类型比较丰富。

2、redis支持主从模式

3、redis支持数据持久化,保证了数据的安全。

4、redis单个value的最大限制是1G,而memcache只有1M

 

redis应用场景:

1、缓存  --缓存,不需要去数据库中去取

2、队列  --冒泡机制

3、数据存储  --直接当做数据库来使用

 

 

 ———— redis的安装————

1、使用yum安装:  <= 建议使用这种安装方式。还有编译安装,参考菜鸟教程或其他

yum install redis

2、安装完成后,使用whereis redis命令查看安装位置,切换到该目录,修改redis.conf,一般在/etc/redis.conf:

主要修改两个:

daemonize yes     //默认不是后台运行,修改成yes即可
port 6379     //默认端口是6379,修改成你想要的端口即可

 3、开启客户端要保证服启动redis服务端:

redis-server /etc/redis.conf

其他的配置项,请查看:http://www.runoob.com/redis/redis-conf.html

 

 

// 登录:

// 服务器上默认直接使用(主机和端口默认,默认没有密码)
redis-cli
// 其他地方登录使用:
redis-cli -h 127.0.0.1 -p 6379 -a password

默认是没有密码的。如果你想要密码登录,在redis.conf文件中配置:

// 1、打开redis.conf:
vim /etc/redis.conf
// 2、添加一行密码设定:
requirepass 123456

 

查看消息:

info    //登陆后查看redis信息

 

// Redis支持五种数据类型:

string(字符串):字符串、整数、浮点。

hash(哈希):是一个键值(key=>value)对集合,key必须是唯一的

list(列表):队列,你可以添加一个元素到列表的头部(左边)或者尾部(右边)。

set(集合):Set是string类型的无序集合,value是唯一的。

zset(sorted set:有序集合):不同的是每个元素都会关联一个double类型的分数。

 

操作示例 - string类型:

SET name "linfeng"    //插入字符串name,值为linfeng
GET name    //获取name的值
DEL name    //删除name

SET string 4   //设置string的值为4(整形)
incr string    //输出为5,此时进行了自增
decrby string 2   //输出为3,此时减了2

 

操作示例 - 队列 - list类型:(最新的在左侧,一个key对应多个val,val可以重复)

lpush list 11    //往队列加入12
lpush list 12    //往队列加入13
lpush list 13    //往队列加入14
lpush list 14    //往队列加入13
lpush list 15    //往队列加入14
 
lpop list    //移出最左侧的15(最后),此时剩下14 13 12 11
rpop list    //输出最右侧的11,此时剩下14 13 12
lrange list    //输出list的数据。为14 13 12
LLEN list   //获取长度,此时为3

 

 操作示例 - 集合 - set:(一个key对应多个val,val不能相同)

sadd set1 1 2 3 4 5 6    //往set1集合里面插入1 2 3 4 5 6
sadd set1 7    //往set1集合里面插入7
sadd set1 7    //返回0,因为已经有7

smembers set1    //输出1 2 3 4 5 6 7 8
scard set1    //输出8,表明有8个元素
sismember set1 8    //返回1,表明此时set1中有8这个元素
sismember set1 9    //返回0,表明此时set2中没有8这个元素

 

 操作示例  - 哈希 - hset:(一个哈希值,对应多个[key => val])

hset hash1 key1 12    //设置hash1的值为[key1=>12]
hset hash1 key2 13    //设置hash1的值为[key2=>13]
hset hash1 key3 13    //设置hash1的值为[key3=>13]
hset hash1 key2 14    //修改hash1的key2的值为14
hget hash1 key1    //输出12

 

操作示例 - 有序集合 - zset:(通过关联一个浮点数为其排序)

zadd zset1 10.1 value1    //设置zset1的值为[10.1=>value1]
zadd zset1 11.1 value2    //设置zset1的值为[11.1=>value2]
zadd zset1 9.2 value3    //设置zset1的值为[9.2=>value3]

zrange zset1 0 2 withscores    //从小到大排序
zrank zset1 value2    //输出2,因为在zset1中排序为第2

zadd zset1 12.1 value3    //把value3的值改为12.1

 

 其他:

keys *    //查看所有的key
DEL key    //删除指定的key

 

 

 ——————————————php操作redis:——————————————

安装前请确保你已经安装了phpize和php-config,使用which命令查看是否安装了拓展

which phpize    //查找phpize的位置
which php-config    //查找php-config的位置
which redis-cli    //查看是否安装了redis

 

如果没有安装的话,请使用命令安装:(我已经安装了,一般都是已经安装了的)

yum install php-devel    //安装php拓展

 

安装PHP redis驱动:

wget https://github.com/phpredis/phpredis/archive/3.1.4.tar.gz      //下载压缩包
tar -zxvf 3.1.4.tar.gz     //解压缩
cd phpredis-3.1.4       //进入 phpredis 目录
which phpize        // 显示安装目录为/alidata/server/php/bin/phpize
/alidata/server/php/bin/phpize        //用phpize生成configure配置文件
which php-config    //查找php-config的位置,显示为/alidata/server/php/bin/php-config
./configure --with-php-config=/alidata/server/php/bin/php-config
make && make install

 安装完成后,出现以下的路径:

/alidata/server/php/lib/php/extensions/no-debug-non-zts-20121212      // => 可以在安装完成的界面看到

 

打开php.ini文件,添加以下内容:

extension_dir = "/alidata/server/php/lib/php/extensions/no-debug-non-zts-20121212/"        //其实我的php.ini文件里面已经有了这个配置,没有的话再添加吧。
extension=redis.so    //没有这个,需要添加

重启nginx和php-fpm:

service nginx restart
service php-fpm restart

——使用phpinfo,可以看到安装好了phpredis

——以上,php下的redis拓展安装好了。

 

各种使用:请参考:http://www.runoob.com/redis/redis-php.html

请注意:在TP5里面使用,如果你没有用到tp5的缓存驱动,也就是没有使用:use think\cache\driver\Redis;

那么你在new Redis的时候,需要加根目录\,比如:

public  function test()
{
    $redis = new \Redis();    //添加根命名空间
    $redis->connect('127.0.0.1', 6379);
    //存储数据到列表中
    $redis->lpush("tutorial-list", "Redis");
    $redis->lpush("tutorial-list", "Mongodb");
    $redis->lpush("tutorial-list", "Mysql");
    // 获取存储的数据并输出
    $arList = $redis->lrange("tutorial-list", 0 ,5);
    print_r($arList);
}

  

——————模拟秒杀代码——————

第一步:往redis中添加50条数据:

// 模拟添加50条数据
public function addTest()
{
    $store = 50;
    $redis = new \Redis();    //添加根命名空间
    $redis->connect('127.0.0.1', 6379);
    $redis->del('goods_store');
    $res = $redis->lLen('goods_store');
    $count = $store - $res;
    //halt($count);
    for ($i = 0; $i < $count; $i++) {
        $redis->lPush('goods_store', 1);  //模拟50个商品库存
    }
    halt($redis->lrange("goods_store", 0, 100));
}

第二步:模拟秒杀代码:

// 秒杀测试
public function test()
{

    $id = input('id');
    if (!$id) {
        Db::table('log')->insert(['addtime' => date('Y-m-d H:i:s'), 'status' => 0]);
        return 'false';
    }
    $redis = new \Redis();    //添加根命名空间
    $redis->connect('127.0.0.1', 6379);
    $count = $redis->lPop('goods_store');
    if (!$count) {
        //无库存
        Db::table('log')->insert(['addtime' => date('Y-m-d H:i:s'), 'status' => 0]);
        return 'false';
    } else {
        // 有库存
        $orderSn = str_rand(20);  //生成20位的随机字符串
        $uid = digital_rand(8);
        $data = Db::table('goods')->field("count,amount")->where("id", $id)->find();  //查找商品
        if (!$data) {
            // 不存在商品
            Db::table('log')->insert(['addtime' => date('Y-m-d H:i:s'), 'status' => 0]);
            return 'false';
        } else {
            Db::table('order')->insert(['order_sn' => $orderSn, 'user_id' => $uid, 'goods_id' => $id, 'price' => $data['amount'], 'status' => 1, 'addtime' => date('Y-m-d H:i:s')]);
            $res = Db::table('goods')->where('id', $id)->setDec('count');
            if ($res) {
                Db::table('log')->insert(['addtime' => date('Y-m-d H:i:s'), 'status' => 1]);
            } else {
                Db::table('log')->insert(['addtime' => date('Y-m-d H:i:s'), 'status' => 0]);
            }
        }
    }
}

使用ab测试工具测试:

ab -n 2000 -c 100 -k http://xxx.xxx.com/backend/login/test/id/1

 

 

——————// 一些常用方法//——————

1,connect

描述:实例连接到一个Redis.

参数:host: string,port: int
返回值:BOOL 成功返回:TRUE;失败返回:FALSE

示例: 

$redis = new \redis();   //请注意,这个是放在特定框架里,需要加跟命名空间。
$result = $redis->connect('127.0.0.1', 6379);  
var_dump($result); //结果:bool(true)

 

2,set描述:设置key和value的值
参数:Key Value
返回值:BOOL 成功返回:TRUE;失败返回:FALSE
示例: 

$result = $redis->set('test',"11111111111");  
var_dump($result);    //结果:bool(true)

 

3,get

描述:获取有关指定键的值
参数:key
返回值:string或BOOL 如果键不存在,则返回 FALSE。否则,返回指定键对应的value值。
范例:

$result = $redis->get('test');  
var_dump($result);   //结果:string(11) "11111111111"

 

 4,delete

描述:删除指定的键
参数:一个键,或不确定数目的参数,每一个关键的数组:key1 key2 key3 … keyN
返回值:删除的项数
范例: 

$redis->set('test',"1111111111111");  
echo $redis->get('test');   //结果:1111111111111  
$redis->delete('test');  
var_dump($redis->get('test'));  //结果:bool(false)

 

5,setnx

描述:如果在数据库中不存在该键,设置关键值参数
参数:key value
返回值:BOOL 成功返回:TRUE;失败返回:FALSE

范例:

$redis->set('test',"1111111111111");  
$redis->setnx('test',"22222222");  
echo $redis->get('test');  //结果:1111111111111  
$redis->delete('test');  
$redis->setnx('test',"22222222");  
echo $redis->get('test');  //结果:22222222

 

6,exists

描述:验证指定的键是否存在
参数key
返回值:Bool 成功返回:TRUE;失败返回:FALSE
范例:

$redis->set('test',"1111111111111");  
var_dump($redis->exists('test'));  //结果:bool(true)

 

7,incr

描述:数字递增存储键值键.
参数:key value:将被添加到键的值
返回值:INT the new value

$redis->set('test',"123");  
var_dump($redis->incr("test"));  //结果:int(124)  
var_dump($redis->incr("test"));  //结果:int(125)

 

8,decr

描述:数字递减存储键值。
参数:key value:将被添加到键的值
返回值:INT the new value
实例:

$redis->set('test',"123");  
var_dump($redis->decr("test"));  //结果:int(122)  
var_dump($redis->decr("test"));  //结果:int(121)

 

 9,getMultiple

描述:取得所有指定键的值。如果一个或多个键不存在,该数组中该键的值为假
参数:其中包含键值的列表数组
返回值:返回包含所有键的值的数组
实例:

$redis->set('test1',"1");  
$redis->set('test2',"2");  
$result = $redis->getMultiple(array('test1','test2'));  
print_r($result);   //结果:Array ( [0] => 1 [1] => 2 )

 

 10,lpush

描述:由列表头部添加字符串值。如果不存在该键则创建该列表。如果该键存在,而且不是一个列表,返回FALSE。
参数:key,value
返回值:成功返回数组长度,失败false
例子:

$redis->delete('test');  
var_dump($redis->lpush("test","111"));   //结果:int(1)  
var_dump($redis->lpush("test","222"));   //结果:int(2)

 

 11,rpush

描述:由列表尾部添加字符串值。如果不存在该键则创建该列表。如果该键存在,而且不是一个列表,返回FALSE。
参数:key,value
返回值:成功返回数组长度,失败false
范例:

$redis->delete('test');  
var_dump($redis->lpush("test","111"));   //结果:int(1)  
var_dump($redis->lpush("test","222"));   //结果:int(2)  
var_dump($redis->rpush("test","333"));   //结果:int(3)  
var_dump($redis->rpush("test","444"));   //结果:int(4)

 

12,lpop

描述:返回和移除列表的第一个元素
参数:key
返回值:成功返回第一个元素的值 ,失败返回false
范例:

$redis->delete('test');  
$redis->lpush("test","111");  
$redis->lpush("test","222");  
$redis->rpush("test","333");  
$redis->rpush("test","444");  
var_dump($redis->lpop("test"));  //结果:string(3) "222"

 

13,lsize,llen

描述:返回的列表的长度。如果列表不存在或为空,该命令返回0。如果该键不是列表,该命令返回FALSE。
参数:Key
返回值:成功返回数组长度,失败false
范例:

$redis->delete('test');  
$redis->lpush("test","111");  
$redis->lpush("test","222");  
$redis->rpush("test","333");  
$redis->rpush("test","444");  
var_dump($redis->lsize("test"));  //结果:int(4)

 

14,lget

描述:返回指定键存储在列表中指定的元素。 0第一个元素,1第二个… -1最后一个元素,-2的倒数第二…错误的索引或键不指向列表则返回FALSE。
参数:key index
返回值:成功返回指定元素的值,失败false
范例:

 

$redis->delete('test');  
$redis->lpush("test","111");  
$redis->lpush("test","222");  
$redis->rpush("test","333");  
$redis->rpush("test","444");  
var_dump($redis->lget("test",3));  //结果:string(3) "444"

 

15,lset

描述:为列表指定的索引赋新的值,若不存在该索引返回false.
参数:key index value
返回值:成功返回true,失败false
范例:

$redis->delete('test');  
$redis->lpush("test","111");  
$redis->lpush("test","222");  
var_dump($redis->lget("test",1));  //结果:string(3) "111"  
var_dump($redis->lset("test",1,"333"));  //结果:bool(true)  
var_dump($redis->lget("test",1));  //结果:string(3) "333"

 

 16,lgetrange

描述:
返回在该区域中的指定键列表中开始到结束存储的指定元素,lGetRange(key, start, end)。0第一个元素,1第二个元素… -1最后一个元素,-2的倒数第二…
参数:key start end
返回值:成功返回查找的值,失败false
范例:

$redis->delete('test');  
$redis->lpush("test","111");  
$redis->lpush("test","222");  
print_r($redis->lgetrange("test",0,-1));  //结果:Array ( [0] => 222 [1] => 111 )

 

 17,lremove

描述:从列表中从头部开始移除count个匹配的值。如果count为零,所有匹配的元素都被删除。如果count是负数,内容从尾部开始删除。
参数:key count value
返回值:成功返回删除的个数,失败false
范例: 

$redis->delete('test');  
$redis->lpush('test','a');  
$redis->lpush('test','b');  
$redis->lpush('test','c');  
$redis->rpush('test','a');  
print_r($redis->lgetrange('test', 0, -1)); //结果:Array ( [0] => c [1] => b [2] => a [3] => a )  
var_dump($redis->lremove('test','a',2));   //结果:int(2)  
print_r($redis->lgetrange('test', 0, -1)); //结果:Array ( [0] => c [1] => b )

 

18,sadd

描述:为一个Key添加一个值。如果这个值已经在这个Key中,则返回FALSE。
参数:key value
返回值:成功返回true,失败false
范例:

 

$redis->delete('test');  
var_dump($redis->sadd('test','111'));   //结果:bool(true)  
var_dump($redis->sadd('test','333'));   //结果:bool(true)  
print_r($redis->sort('test')); //结果:Array ( [0] => 111 [1] => 333 )

 

19,sremove

描述:删除Key中指定的value值
参数:key member
返回值:true or false
范例:

$redis->delete('test');  
$redis->sadd('test','111');  
$redis->sadd('test','333');  
$redis->sremove('test','111');  
print_r($redis->sort('test'));    //结果:Array ( [0] => 333 )

 

20,smove

描述:将Key1中的value移动到Key2中
参数:srcKey dstKey member
返回值:true or false
范例

$redis->delete('test');  
$redis->delete('test1');  
$redis->sadd('test','111');  
$redis->sadd('test','333');  
$redis->sadd('test1','222');  
$redis->sadd('test1','444');  
$redis->smove('test',"test1",'111');  
print_r($redis->sort('test1'));    //结果:Array ( [0] => 111 [1] => 222 [2] => 444 )

 

21,scontains

描述:检查集合中是否存在指定的值。
参数:key value
返回值:true or false
范例:

$redis->delete('test');  
$redis->sadd('test','111');  
$redis->sadd('test','112');  
$redis->sadd('test','113');  
var_dump($redis->scontains('test', '111')); //结果:bool(true)

 

22,ssize

描述:返回集合中存储值的数量
参数:key
返回值:成功返回数组个数,失败0
范例:

$redis->delete('test');  
$redis->sadd('test','111');  
$redis->sadd('test','112');  
echo $redis->ssize('test');   //结果:2

 

23,spop

描述:随机移除并返回key中的一个值
参数:key
返回值:成功返回删除的值,失败false
范例:

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
var_dump($redis->spop("test"));  //结果:string(3) "333"

 

 24,sinter

描述:返回一个所有指定键的交集。如果只指定一个键,那么这个命令生成这个集合的成员。如果不存在某个键,则返回FALSE。
参数:key1, key2, keyN
返回值:成功返回数组交集,失败false
范例:

 

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
$redis->sadd("test1","111");  
$redis->sadd("test1","444");  
var_dump($redis->sinter("test","test1"));  //结果:array(1) { [0]=> string(3) "111" }

 

25,sinterstore

描述:执行sInter命令并把结果储存到新建的变量中。
参数:
Key: dstkey, the key to store the diff into.
Keys: key1, key2… keyN. key1..keyN are intersected as in sInter.
返回值:成功返回,交集的个数,失败false
范例:

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
$redis->sadd("test1","111");  
$redis->sadd("test1","444");  
var_dump($redis->sinterstore('new',"test","test1"));  //结果:int(1)  
var_dump($redis->smembers('new'));  //结果:array(1) { [0]=> string(3) "111" }

 

26,sunion

描述:
返回一个所有指定键的并集
参数:
Keys: key1, key2, … , keyN
返回值:成功返回合并后的集,失败false
范例:

 

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
$redis->sadd("test1","111");  
$redis->sadd("test1","444");  
print_r($redis->sunion("test","test1"));  //结果:Array ( [0] => 111 [1] => 222 [2] => 333 [3] => 444 )

 

27,sunionstore

描述:执行sunion命令并把结果储存到新建的变量中。
参数:
Key: dstkey, the key to store the diff into.
Keys: key1, key2… keyN. key1..keyN are intersected as in sInter.
返回值:成功返回,交集的个数,失败false
范例:

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
$redis->sadd("test1","111");  
$redis->sadd("test1","444");  
var_dump($redis->sinterstore('new',"test","test1"));  //结果:int(4)  
print_r($redis->smembers('new'));  //结果:Array ( [0] => 111 [1] => 222 [2] => 333 [3] => 444 )

 

28,sdiff

描述:返回第一个集合中存在并在其他所有集合中不存在的结果
参数:Keys: key1, key2, … , keyN: Any number of keys corresponding to sets in redis.
返回值:成功返回数组,失败false
范例:

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
$redis->sadd("test1","111");  
$redis->sadd("test1","444");  
print_r($redis->sdiff("test","test1"));  //结果:Array ( [0] => 222 [1] => 333 )

 

29,sdiffstore

描述:执行sdiff命令并把结果储存到新建的变量中。
参数:
Key: dstkey, the key to store the diff into.
Keys: key1, key2, … , keyN: Any number of keys corresponding to sets in redis
返回值:成功返回数字,失败false
范例:

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
$redis->sadd("test","333");  
$redis->sadd("test1","111");  
$redis->sadd("test1","444");  
var_dump($redis->sdiffstore('new',"test","test1"));  //结果:int(2)  
print_r($redis->smembers('new'));  //结果:Array ( [0] => 222 [1] => 333 )

 

30,smembers, sgetmembers

描述:
返回集合的内容
参数:Key: key
返回值:An array of elements, the contents of the set.
范例:

$redis->delete('test');  
$redis->sadd("test","111");  
$redis->sadd("test","222");  
print_r($redis->smembers('test'));  //结果:Array ( [0] => 111 [1] => 222 )

 

php-redis当中,有很多不同名字,但是功能一样的函数,例如:lrem和lremove,这里就不例举了。