什么是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,这里就不例举了。