= =琢磨了一下 hiredis 感觉很赞。
用C++封装了一下。
/**
* 缓存工具
* Created on : 2014-05-08
* Author : frodo
*/
#ifndef FSCACHEDB_H_
#define FSCACHEDB_H_
#include /* for size_t */
#include /* for va_list */
class FSCacheData{
public:
virtual bool isNil(int index=0) = 0;
virtual int getInt(int index=0) = 0;
virtual float getFloat(int index=0) = 0;
virtual const char* getString(int index=0) = 0;
virtual bool getBool(int index=0) = 0;
virtual char* getBinary(int index=0) = 0;
virtual size_t getLength(int index=0) = 0;
virtual size_t getArrayLength() = 0;
virtual bool isArray() = 0;
};
class FSCacheConfig{
};
class FSCacheDB{
public:
FSCacheDB();
public:
/**
* 设置一个字符串到key
* @key : key
* @value : value
* @return : 是否成功
*/
virtual bool setString(const char* key, const char* value, unsigned int expire=0) = 0;
/**
* 设置一个二进制数据到key
* @key : key
* @value : value
* @return : 是否成功
*/
virtual bool setBinary(const char* key, unsigned char* data, size_t len) = 0;
/**
* 设置一个字符串到hash
* @hash_name : hahs名字
* @key : hash_key
* @value : value
* @return : 是否成功
*/
virtual bool setHashString(const char* hash_name, const char* key, const char* value) = 0;
/**
* 设置一个二进制数据到hash
* @hash_name : hash名字
* @key : hash_key
* @value : value
* @len : len
* @return : 是否成功
*/
virtual bool setHashBinary(const char* hash_name, const char* key, unsigned char* data, size_t len) = 0;
/**
* 添加一个字符串到列表尾
* @list_name : 列表名字
* @value : 数据
* @return : 返回数组长度
*/
virtual int pushStringToList(const char* list_name, const char* value) = 0;
/**
* 插入一个二进制数据到列表
* @list_name : 列表名字
* @value : 数据
* @return : 返回数组长度
*/
virtual bool setBinaryToList(const char* list_name, int index, char* data, size_t len) = 0;
/**
* 插入一个字符串到列表
* @list_name : 列表名字
* @value : 数据
* @return : 返回数组长度
*/
virtual bool setStringToList(const char* list_name, int index, const char* value) = 0;
/**
* 添加一个二进制数据到列表尾
* @list_name : 列表名字
* @value : 数据
* @return : 返回数组长度
*/
virtual int pushBinaryToList(const char* list_name, char* binary, size_t len) = 0;
/**
* 添加一个数据到有序集合中
* @set_name : 集合名字
* @score : 排序值
* @value : 该排序对应的数据
*/
virtual void addSortedSet(const char* set_name, int score, const char* value)=0;
/**
* 根据对应的数据得到一个排名
* @set_name : 集合名字
* @value : 获取该数据的排名
* @return : 返回排名
*/
virtual int getSortByString(const char* set_name, const char* value)=0;
public:
/**
* 根据key得到一个value
* @key : key_name
* @outdata : 输出数据
* @return : 是否成功
*/
virtual bool getDataFromKey(const char* key, FSCacheData* outdata) = 0;
/**
* 获取hash内的value
* @hash_name : hash_name
* @key : key
* @outdata : 输出数据
* @return : 是否成功
*/
virtual bool getDataFromHash(const char* hash_name, const char* key, FSCacheData* outdata) = 0;
/**
* 从有序集合中得到数据
* @set_name : set_name
* @starr : start
* @end : end
* @outdata : 输出数据
* @return : 是否成功
*/
virtual bool getDataFromSoctedSet(const char* set_name, size_t start, size_t end, FSCacheData* outdata) = 0;
/**
* 从列表中获取数据
* @list_name : list_name
* @starr : start
* @end : end
* @outdata : 输出数据
* @return : 是否成功
*/
virtual bool getDataFromList(const char* list_name, size_t start, size_t end, FSCacheData* outdata) = 0;
/**
* 从列表长度
* @list_name : list_name
* @return : 长度
*/
virtual int getListLen(const char* list_name) = 0;
/**
* 弹出list的头部并且返回
* @list_name : list_name
* @outdata : 输出数据
* @return : 是否成功
*/
virtual bool shiftDataFromList(const char* list_name, FSCacheData* outdata) = 0;
/**
* 移除list的尾部并且返回
* @list_name : list_name
* @outdata : 输出数据
* @return : 是否成功
*/
virtual bool popDataFromList(const char* list_name, FSCacheData* outdata) = 0;
public:
/**
* 获取一个自增的值
* @key : 自增key
* @return : 返回自增后的数值
*/
virtual int incrKey(const char* key) = 0;
/**
* 删除一个key
* @key : key
* @value : value
* @return : 是否成功
*/
virtual bool delKey(const char* key) = 0;
public:
/**
* 选择DB
* @index : index 0~16
*/
virtual bool selectDB(int index) = 0;
/**
* 获取当前使用中的DB
* @return : 返回使用中的DB
*/
virtual int getDBNumber() = 0;
public:
/**
* 初始化链接
*/
virtual bool connect(FSCacheConfig* conf)=0;
/**
* 关闭链接
*/
virtual bool close()=0;
static FSCacheDB* sharedCacheDB();
};
#endif /* FSCACHEDB_H_ */
#include "FSCacheDB.h"
static FSCacheDB* s_sharedCache = NULL;
FSCacheDB::FSCacheDB(){
s_sharedCache = this;
}
FSCacheDB* FSCacheDB::sharedCacheDB(){
return s_sharedCache;
}
/*
* FSRedis.h
*
* Created on: 2014-5-8
* Author: frodo
*/
#ifndef FSREDIS_H_
#define FSREDIS_H_
#include "FSCacheDB.h"
#include
class FSRedis;
class FSRedisData : public FSCacheData
{
private:
redisReply* m_pReply;
friend class FSRedis;
void initReply(redisReply*);
private:
bool checkType(int index);
public:
FSRedisData();
~FSRedisData();
private:
redisReply* getReply(int index=0);
public:
virtual bool isNil(int index=0) ;
virtual int getInt(int index=0) ;
virtual float getFloat(int index=0) ;
virtual const char* getString(int index=0) ;
virtual bool getBool(int index=0) ;
virtual char* getBinary(int index=0) ;
virtual size_t getLength(int index=0);
virtual bool isArray();
virtual size_t getArrayLength();
};
class FSRedisConfig : public FSCacheConfig{
public:
const char* host;
int port;
int db_index;
struct timeval timeout;
};
class FSRedis : public FSCacheDB{
public:
FSRedis();
virtual ~FSRedis();
private:
int m_iSelectedDB;
redisContext* m_pContext;
public:
virtual bool setString(const char* key, const char* value, unsigned int expire=0) ;
virtual bool setBinary(const char* key, unsigned char* data, size_t len) ;
virtual bool setHashString(const char* hash_name, const char* key, const char* value) ;
virtual bool setHashBinary(const char* hash_name, const char* key, unsigned char* data, size_t len) ;
virtual int pushStringToList(const char* list_name, const char* value);
virtual int pushBinaryToList(const char* list_name, char* binary, size_t len);
virtual bool setBinaryToList(const char* list_name, int index, char* data, size_t len) ;
virtual bool setStringToList(const char* list_name, int index, const char* value) ;
virtual void addSortedSet(const char* set_name, int score, const char* value);
virtual int getSortByString(const char* set_name, const char* value);
public:
virtual int getListLen(const char* list_name) ;
virtual bool getDataFromList(const char* list_name, size_t start, size_t end, FSCacheData* outdata);
virtual bool getDataFromSoctedSet(const char* set_name, size_t start, size_t end, FSCacheData* outdata);
virtual bool getDataFromKey(const char* key, FSCacheData* outdata) ;
virtual bool getDataFromHash(const char* hash_name, const char* key, FSCacheData* outdata) ;
virtual bool shiftDataFromList(const char* list_name, FSCacheData* outdata) ;
virtual bool popDataFromList(const char* list_name, FSCacheData* outdata) ;
public:
virtual int incrKey(const char* key) ;
virtual bool delKey(const char* key) ;
public:
virtual bool selectDB(int index);
virtual int getDBNumber();
public:
bool connect(FSCacheConfig* conf);
bool close();
public: // redis
bool checkReplay(redisReply*);
bool executeCommand(const char* command, ...);
bool executeCommand(const char* command, va_list ap);
};
#endif /* FSREDIS_H_ */
/*
* FSRedis.cpp
*
* Created on: 2014-5-8
* Author: frodo
*/
#include "FSRedis.h"
#include
#include
#define _CHECK_CONNECT if(!m_pContext) return false;
FSRedisData::FSRedisData():m_pReply(NULL){
}
FSRedisData::~FSRedisData(){
if(m_pReply){
freeReplyObject(m_pReply);
}
}
redisReply* FSRedisData::getReply(int index){
if(m_pReply->type == REDIS_REPLY_ARRAY){
return m_pReply->element[index];
}else{
return m_pReply;
}
}
void FSRedisData::initReply(redisReply* reply){
if(m_pReply){
freeReplyObject(m_pReply);
m_pReply = NULL;
}
m_pReply = reply;
}
bool FSRedisData::checkType(int index){
return !isNil(index) &&
(getReply(index)->type == REDIS_REPLY_STRING ||
getReply(index)->type == REDIS_REPLY_INTEGER ||
getReply(index)->type == REDIS_REPLY_STATUS);
}
bool FSRedisData::isArray(){
return m_pReply->type == REDIS_REPLY_ARRAY;
}
bool FSRedisData::isNil(int index){
return !m_pReply || m_pReply->type == REDIS_REPLY_NIL || m_pReply->type == REDIS_REPLY_ERROR;
}
int FSRedisData::getInt(int index) {
if(!checkType(index)){
return 0;
}
return atoi(getString());
}
float FSRedisData::getFloat(int index) {
if(!checkType(index)){
return 0;
}
return atof(getString());
}
const char* FSRedisData::getString(int index) {
if(!checkType(index)){
return 0;
}
return getReply(index)->str;
}
bool FSRedisData::getBool(int index) {
if(!checkType(index)){
return 0;
}
return strcasecmp("true", getReply(index)->str) == 0;
}
char* FSRedisData::getBinary(int index) {
return getReply(index)->str;
}
size_t FSRedisData::getLength(int index){
return getReply(index)->len;
}
size_t FSRedisData::getArrayLength(){
if(m_pReply){
return m_pReply->elements;
}
return 1;
}
FSRedis::FSRedis() {
}
FSRedis::~FSRedis() {
this->close();
}
bool FSRedis::connect(FSCacheConfig* conf){
FSRedisConfig* config = (FSRedisConfig*)conf;
m_pContext = redisConnectWithTimeout(config->host, config->port, config->timeout);
if(!m_pContext || m_pContext->err){
printf("init redis is erro code[%d], msg[%s]\n", m_pContext->err, m_pContext->errstr);
goto fail;
}
if(!selectDB(config->db_index)){
goto fail;
}
goto success;
success:
return true;
fail:
return false;
}
bool FSRedis::close(){
if(m_pContext){
redisFree(this->m_pContext);
m_pContext = NULL;
return true;
}
return false;
}
bool FSRedis::selectDB(int index){
if( executeCommand("SELECT %d", index) ){
m_iSelectedDB = index;
return true;
}
return false;
}
int FSRedis::getDBNumber(){
return m_iSelectedDB;
}
bool FSRedis::checkReplay(redisReply* replay){
return replay != NULL;
}
bool FSRedis::executeCommand(const char* command, ...){
va_list ap;
va_start(ap,command);
bool ret = executeCommand(command, ap);
va_end(ap);
return ret;
}
bool FSRedis::executeCommand(const char* command, va_list ap){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, command, ap);
if(!checkReplay(reply)){
return false;
}
freeReplyObject(reply);
return true;
}
bool FSRedis::setString(const char* key, const char* value, unsigned int expire){
_CHECK_CONNECT
redisReply* reply = NULL;
if(expire != 0){
reply = (redisReply*)redisCommand(m_pContext, "SETEX %s %d %s", key, expire, value);
}else{
reply = (redisReply*)redisCommand(m_pContext, "SET %s %s", key, value);
}
if(!checkReplay(reply)){
return false;
}
freeReplyObject(reply);
return true;
}
bool FSRedis::setBinary(const char* key, unsigned char* data, size_t len){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "SET %s %b", key, data, len);
if(!checkReplay(reply)){
return false;
}
freeReplyObject(reply);
return true;
}
bool FSRedis::setHashString(const char* hash_name, const char* key, const char* value){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "HSET %s %s %s", hash_name, key, value);
if(!checkReplay(reply)){
return false;
}
freeReplyObject(reply);
return true;
}
bool FSRedis::setHashBinary(const char* hash_name, const char* key, unsigned char* data, size_t len){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "HSET %s %s %b", hash_name, key, data, len);
if(!checkReplay(reply)){
return false;
}
freeReplyObject(reply);
return true;
}
bool FSRedis::setBinaryToList(const char* list_name, int index, char* data, size_t len) {
_CHECK_CONNECT
redisReply* reply = (redisReply*) redisCommand(m_pContext, "LSET %s %d %b", list_name, index, data, len);
if (!checkReplay(reply)) {
return false;
}
freeReplyObject(reply);
return true;
}
bool FSRedis::setStringToList(const char* list_name, int index, const char* value) {
_CHECK_CONNECT
redisReply* reply = (redisReply*) redisCommand(m_pContext, "LSET %s %d %s", list_name, index, value);
if(!checkReplay(reply)){ return false; }
freeReplyObject(reply);
return true;
}
int FSRedis::pushStringToList(const char* list_name, const char* value){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "RPUSH %s %s", list_name, value);
if(!checkReplay(reply)){
return -1;
}
int len = reply->integer;
freeReplyObject(reply);
return len;
}
int FSRedis::pushBinaryToList(const char* list_name, char* binary, size_t len){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "RPUSH %s %b", list_name, binary, len);
if(!checkReplay(reply)){
return -1;
}
int ret = reply->integer;
freeReplyObject(reply);
return ret;
}
void FSRedis::addSortedSet(const char* set_name, int score, const char* value){
if(!m_pContext) return;
redisReply* reply = (redisReply*)redisCommand(m_pContext, "ZADD %s %d %s", set_name, score, value);
freeReplyObject(reply);
}
int FSRedis::getSortByString(const char* set_name, const char* value){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "ZRANK %s %s", set_name, value);
if(!checkReplay(reply)){
return -1;
}
int ret = reply->integer;
freeReplyObject(reply);
return ret;
}
bool FSRedis::getDataFromKey(const char* key, FSCacheData* outdata){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "GET %s", key);
if(!checkReplay(reply)){ return false; }
((FSRedisData*)outdata)->initReply(reply);
return true;
}
bool FSRedis::getDataFromHash(const char* hash_name, const char* key, FSCacheData* outdata){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "HGET %s %s", hash_name, key);
if(!checkReplay(reply)){ return false; }
((FSRedisData*)outdata)->initReply(reply);
return true;
}
int FSRedis::getListLen(const char* list_name) {
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "LLEN %s", list_name);
if(!checkReplay(reply)){ return 0; }
int ret = reply->integer;
freeReplyObject(reply);
return ret;
}
bool FSRedis::getDataFromList(const char* list_name, size_t start, size_t end, FSCacheData* outdata){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "LRANGE %s %d %d", list_name, start, end);
if(!checkReplay(reply)){ return false; }
((FSRedisData*)outdata)->initReply(reply);
return true;
}
bool FSRedis::getDataFromSoctedSet(const char* set_name, size_t start, size_t end, FSCacheData* outdata){
_CHECK_CONNECT
redisReply* reply = (redisReply*) redisCommand(m_pContext, "ZRANGE %s %d %d", set_name, start, end);
if (!checkReplay(reply)) {
return false;
}
((FSRedisData*) outdata)->initReply(reply);
return true;
}
bool FSRedis::shiftDataFromList(const char* list_name, FSCacheData* outdata){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "LPOP %s %s", list_name);
if(!checkReplay(reply)){ return false; }
((FSRedisData*)outdata)->initReply(reply);
return true;
}
bool FSRedis::popDataFromList(const char* list_name, FSCacheData* outdata){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "RPOP %s %s", list_name);
if(!checkReplay(reply)){ return false; }
((FSRedisData*)outdata)->initReply(reply);
return true;
}
int FSRedis::incrKey(const char* key){
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "INCR %s", key);
if(!checkReplay(reply)) return false;
int ret = reply->integer;
freeReplyObject(reply);
return ret;
}
bool FSRedis::delKey(const char* key) {
_CHECK_CONNECT
redisReply* reply = (redisReply*)redisCommand(m_pContext, "DEL %s", key);
if(!checkReplay(reply)) return false;
int ret = reply->integer;
freeReplyObject(reply);
return ret > 0;
}
test
#include
#include
#include
#include "FSOutputStream.h"
#include "FSInputStream.h"
#include
#include "FSRedis.h"
using namespace std;
void test(){
FSRedisData data;
FSRedisConfig conf;
conf.db_index = 0;
conf.host = "127.0.0.1";
conf.port = 6379;
conf.timeout.tv_sec = 3;
conf.timeout.tv_usec = 0;
FSRedis db;
db.connect(&conf);
db.setString("MYNAME", "FRODO");
db.setString("MYLOVE", "QIAO");
db.getDataFromKey("MYNAME", &data);
printf("%s\n", data.getString());
db.getDataFromKey("MYLOVE", &data);
printf("%s\n", data.getString());
FSOutputStream fos;
for(int i = 0 ; i < 50 ; i++){
fos.writeInt32(i);
}
db.setBinary("IDS", fos.getData(), fos.getLength());
db.getDataFromKey("IDS", &data);
FSInputStream fis(data.getBinary(), data.getLength());
for(int i = 0 ; i< 50 ; i++){
printf("%d \n", fis.readUInt32());
}
if(db.getDataFromKey("MYLOVE", &data)){
printf("%s\n", data.getString());
}else{
printf("get data fail\n");
}
}
int main() {
test();
return 0;
}