>  기사  >  백엔드 개발  >  Redis에서 비트맵의 놀라운 사용

Redis에서 비트맵의 놀라운 사용

高洛峰
高洛峰원래의
2017-02-09 10:30:321440검색

Redis에서는 set, get 및 기타 명령을 자주 사용합니다. 주의 깊게 살펴보면 setbit 및 getbit라는 유사한 명령이 여러 개 있다는 것을 눈치채셨나요?

BitMap이란?

비트를 사용하여 요소에 해당하는 값이나 상태를 나타내며, 키는 해당 요소 자체입니다. 우리는 8비트가 바이트를 형성할 수 있다는 것을 알고 있으므로 비트맵 자체가 저장 공간을 크게 절약할 수 있습니다.

Redis의 BitMap

Redis는 버전 2.2.0부터 setbit, getbit, bitcount 등 여러 비트맵 관련 명령을 추가했습니다. 새로운 명령이지만 setbit와 같은 명령은 단지 set의 확장이므로 새로운 데이터 유형이 추가되지 않습니다.

setbit 명령 소개

명령SETBIT key offset value
복잡성O(1)
오프셋(0 또는 1만)에서 키의 비트 값(문자열)을 설정하거나 지웁니다.

공간 점유와 처음으로 공간을 할당하는 데 필요한 시간

2010년 MacBook Pro에서 오프셋은 2^32-1(512MB 할당)이 약 300ms가 걸리며, 오프셋은 2^ 30-1(할당된 128MB)은 ~80ms가 걸리고, 오프셋 2^28-1(할당된 32MB)은 ~30ms가 걸리고, 오프셋 2^26-1(할당된 8MB)은 8ms가 걸립니다. <공식 문서에서>
대략적인 공간 점유 계산 공식은 다음과 같습니다. ($offset/8/1024/1024)MB

사용 시나리오 1: 사용자 로그인

많은 웹사이트에서 로그인 기능을 제공합니다(고려되지 않음). 여기) 데이터 구현이 중요합니다), 지난 달의 체크인 상태를 표시해야 합니다. 비트맵을 사용하는 경우 어떻게 해야 합니까? 코드는 한 단어로 공개됩니다!

<?php
$redis = new Redis();
$redis->connect('127.0.0.1');


//用户uid
$uid = 1;

//记录有uid的key
$cacheKey = sprintf("sign_%d", $uid);

//开始有签到功能的日期
$startDate = '2017-01-01';

//今天的日期
$todayDate = '2017-01-21';

//计算offset
$startTime = strtotime($startDate);
$todayTime = strtotime($todayDate);
$offset = floor(($todayTime - $startTime) / 86400);

echo "今天是第{$offset}天" . PHP_EOL;

//签到
//一年一个用户会占用多少空间呢?大约365/8=45.625个字节,好小,有木有被惊呆?
$redis->setBit($cacheKey, $offset, 1);

//查询签到情况
$bitStatus = $redis->getBit($cacheKey, $offset);
echo 1 == $bitStatus ? '今天已经签到啦' : '还没有签到呢';
echo PHP_EOL;

//计算总签到次数
echo $redis->bitCount($cacheKey) . PHP_EOL;

/**
* 计算某段时间内的签到次数
* 很不幸啊,bitCount虽然提供了start和end参数,但是这个说的是字符串的位置,而不是对应"位"的位置
* 幸运的是我们可以通过get命令将value取出来,自己解析。并且这个value不会太大,上面计算过一年一个用户只需要45个字节
* 给我们的网站定一个小目标,运行30年,那么一共需要1.31KB(就问你屌不屌?)
*/
//这是个错误的计算方式
echo $redis->bitCount($cacheKey, 0, 20) . PHP_EOL;

사용 시나리오 2: 활성 사용자 수 계산

시간을 캐시 키로 사용하고 해당 날짜에 활성 상태인 경우 사용자 ID를 1로 설정
그렇다면 어떻게 해야 합니까? 특정 날짜를 계산하나요? /month/year의 활성 사용자는 (당분간 통계 기간 내 온라인 시간 중 하루만 활성이라고 동의함) 다음 redis 명령
명령을 사용하세요. BITOP operation destkey key [key ...]
설명: 하나 이상의 바이너리 문자열 키에 대해 비트 연산을 수행하고 결과를 destkey에 저장합니다.
참고: BITOP 명령은 AND, OR, NOT 및 /1024=6MB의 네 가지 작업 중 모든 매개변수를 지원합니다

사용 시나리오 3: 사용자 온라인 상태

프로젝트를 개발했습니다. 얼마 전 상대방이 현재 사용자가 온라인 상태인지 조회할 수 있는 인터페이스를 제공했습니다. 상대방이 어떻게 하는지 몰라서 직접 생각해봤는데, 비트맵을 사용하는 것이 공간을 절약하고 효율적인 방법이고, 키가 하나만 있으면 되고, 온라인이면 사용자 ID가 오프셋됩니다. 는 1로 설정되고, 온라인이 아닌 경우에는 1.0으로 설정됩니다. 위의 시나리오와 마찬가지로 5000W 사용자에게는 6MB의 공간만 필요합니다.

아아아아

사실 BitMap은 다양한 시나리오에서 사용될 수 있고(물론 일부 제한이 있을 수 있음) 아이디어가 계속 퍼질 수 있습니다~ 친구들이 저에게 메시지를 남겨서 토론할 수 있습니다~

Redis에서는 set 및 get과 같은 명령이 자주 사용됩니다. 주의 깊게 살펴보면 setbit 및 getbit라는 유사한 명령이 여러 개 있다는 것을 알고 계셨습니까?

BitMap이란?

은 요소의 값이나 상태를 비트를 통해 나타내며, 키는 해당 요소 자체입니다. 우리는 8비트가 바이트를 형성할 수 있다는 것을 알고 있으므로 비트맵 자체가 저장 공간을 크게 절약할 수 있습니다.

Redis의 BitMap

Redis는 버전 2.2.0부터

,

, setbit 등 여러 비트맵 관련 명령을 추가했습니다. 새로운 명령이지만 getbit와 같은 명령은 단지 bitcount의 확장이므로 새로운 데이터 유형이 추가되지 않습니다. setbitsetsetbit 명령 소개

명령

복잡성SETBIT key offset value
오프셋(0 또는 1만)에서 키의 비트 값(문자열)을 설정하거나 지웁니다. O(1)
공간 점유와 처음으로 공간을 할당하는 데 필요한 시간

2010년 MacBook Pro에서 오프셋은 2^32-1(512MB 할당)이 약 300ms가 걸리며, 오프셋은 2^ 30-1(할당된 128MB)은 ~80ms가 걸리고, 오프셋 2^28-1(할당된 32MB)은 ~30ms가 걸리고, 오프셋 2^26-1(할당된 8MB)은 8ms가 걸립니다. <공식 문서에서>

대략적인 공간 점유 계산 공식은 다음과 같습니다.


($offset/8/1024/1024)MB사용 시나리오 1: 사용자 로그인

많은 웹사이트에서 로그인 기능을 제공합니다(고려되지 않음). 여기) 데이터 구현이 중요합니다), 지난 달의 체크인 상태를 표시해야 합니다. 비트맵을 사용하는 경우 어떻게 해야 합니까? 코드는 한 단어로 공개됩니다!

<?php
$redis = new Redis();
$redis->connect('127.0.0.1');


//用户uid
$uid = 1;

//记录有uid的key
$cacheKey = sprintf("sign_%d", $uid);

//开始有签到功能的日期
$startDate = '2017-01-01';

//今天的日期
$todayDate = '2017-01-21';

//计算offset
$startTime = strtotime($startDate);
$todayTime = strtotime($todayDate);
$offset = floor(($todayTime - $startTime) / 86400);

echo "今天是第{$offset}天" . PHP_EOL;

//签到
//一年一个用户会占用多少空间呢?大约365/8=45.625个字节,好小,有木有被惊呆?
$redis->setBit($cacheKey, $offset, 1);

//查询签到情况
$bitStatus = $redis->getBit($cacheKey, $offset);
echo 1 == $bitStatus ? '今天已经签到啦' : '还没有签到呢';
echo PHP_EOL;

//计算总签到次数
echo $redis->bitCount($cacheKey) . PHP_EOL;

/**
* 计算某段时间内的签到次数
* 很不幸啊,bitCount虽然提供了start和end参数,但是这个说的是字符串的位置,而不是对应"位"的位置
* 幸运的是我们可以通过get命令将value取出来,自己解析。并且这个value不会太大,上面计算过一年一个用户只需要45个字节
* 给我们的网站定一个小目标,运行30年,那么一共需要1.31KB(就问你屌不屌?)
*/
//这是个错误的计算方式
echo $redis->bitCount($cacheKey, 0, 20) . PHP_EOL;

사용 시나리오 2: 활성 사용자 계산

시간을 캐시 키로 사용하고 해당 날짜에 사용자 ID가 오프셋되면 1로 설정
그럼 어떻게 계산해야 합니까? 일/월/ 해당 연도의 활성 사용자(당분간 통계 기간 내 하루만 온라인으로 활성 상태라고 합의함)는 다음 redis 명령
명령 BITOP operation destkey key [key ...] 설명: 하나 이상의 바이너리 비트를 저장하려면 문자열 키에 대해 비트 연산을 수행하고 결과를 destkey에 저장합니다.
참고: BITOP 명령은 AND, OR, NOT 및 /1024=6MB의 네 가지 작업 매개 변수를 지원합니다.

사용 시나리오 3: 사용자 온라인 상태

프로젝트를 개발했습니다. 얼마 전 상대방이 현재 사용자가 온라인 상태인지 조회할 수 있는 인터페이스를 제공했습니다. 상대방이 어떻게 하는지 몰라서 직접 생각해봤는데, 비트맵을 사용하는 것이 공간을 절약하고 효율적인 방법이고, 키가 하나만 있으면 되고, 온라인이면 사용자 ID가 오프셋됩니다. 는 1로 설정되고, 온라인이 아닌 경우에는 1.0으로 설정됩니다. 위의 시나리오와 마찬가지로 5000W 사용자에게는 6MB의 공간만 필요합니다.

아아아아

사실 BitMap은 다양한 시나리오에서 사용될 수 있고(물론 일부 제한이 있을 수 있음) 아이디어가 계속 퍼질 수 있습니다~ 친구들이 저에게 메시지를 남겨서 토론할 수 있습니다~

Redis에서 비트맵을 더욱 다양하게 활용하는 방법 관련 기사는 PHP 중국어 웹사이트를 참고하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.