redis技巧--IP地址查询对应城市
场景:
根据IP地址判断用户所在地,虽然网上有好多篇了,但我记录一个一看就懂的,不用看超长文字再自己理解了。
我们有城市和IP地址段的对应关系,如:
上海: 202.127.0.0 ~ 202.127.4.255
北京: 122.200.64.0 ~ 122.207.255.255
此时如果用户的IP地址为122.202.2.0,我们就能根据这个表知道他的地址位于北京。
方法:
使用有序集合记录这个表,每个城市存储两条数据,一条是城市名对应的分数为最大IP地址。另一条是“#”加上城市名对应的分数为最小IP地址。
分析:
取大于当前IP的最近的上一个分数值,如果值是#开头,说明是某一个城市的最小IP,就是当前IP不在任何城市范围内,如果不是#开头,那IP就属于这个城市。
|----上海最大IP(80=>sh)
|----上海最小IP(50=>#sh)
|----北京最大IP(30=>bj)
|----北京最小IP(20=>#bj)
代码实现:
//先用ip2long把IP地址转为对应的十进制数,这里为了方便直接使用虚拟数字表示
$redis->zAdd('zset', 20, '#bj'); //最小IP加#
$redis->zAdd('zset', 30, 'bj');
$redis->zAdd('zset', 50, '#sh'); //最小IP加#
$redis->zAdd('zset', 80, 'sh');
//获取IP对应的城市
$cityIp = 34;
$b = $redis->zRangeByScore('zset', $cityIp, '+inf', array( 'limit'=>array(0,1) ));
if ($b) {
if (strpos($b[0], '#') === 0) {
echo '城市不在';
} else {
echo '城市是',print_r($b);
}
} else {
echo '城市不存在';
}