hiRedis基礎(chǔ)教程

本文只總結(jié)常規(guī)會用到的hiRedis使用方法袜炕,一般順序?yàn)橄扔?redisConnect 連接數(shù)據(jù)庫,然后用 redisCommand 執(zhí)行命令犁罩,執(zhí)行完后用 freeReplyObject 來釋放redisReply對象,最后用 redisFree 來釋放整個連接。

hiredis直接去git上克隆氏身,地址:https://github.com/redis/hiredis,然后make && make install

連接

函數(shù)原型

redisContext *redisConnect(const char *ip, int port);
void redisFree(redisContext *c);

使用示例

//連接redis惑畴,若出錯redisContext.err會設(shè)置為1蛋欣,redisContext.errstr會包含描述錯誤信息
redisContext *redis_handle = redisConnect("192.163.122.1", 8889);
if (redis_handle == NULL || redis_handle->err) {
    if (redis_handle) {
        printf("Error:%s\n", redis_handle->errstr);
    } else {
        printf("can't allocate redis context\n");
    }
    return -1;
}
//do your work
redisFree(redis_handle);

執(zhí)行Redis命令

函數(shù)原型

typedef struct redisReply {
    int type; /* REDIS_REPLY_* */
    long long integer; /* The integer when type is REDIS_REPLY_INTEGER */
    int len; /* Length of string */
    char *str; /* Used for both REDIS_REPLY_ERROR and REDIS_REPLY_STRING */
    size_t elements; /* number of elements, for REDIS_REPLY_ARRAY */
    struct redisReply **element; /* elements vector for REDIS_REPLY_ARRAY */
} redisReply;

void *redisCommand(redisContext *c, const char *format, ...);
void freeReplyObject(void *reply);

redisReply結(jié)構(gòu)的type定義如下

  • REDIS_REPLY_STATUS:
    表示返回了一個狀態(tài)。狀態(tài)字符串可以使用reply->str來訪問如贷。該字符串的長度可以使用reply->len來訪問陷虎。例如set成功,會返回狀態(tài)杠袱,str為"OK"
  • REDIS_REPLY_ERROR:
    表示執(zhí)行錯誤尚猿。錯誤字符串可以使用reply->str來訪問
  • REDIS_REPLY_INTEGER:
    表示返回一個整數(shù)。整數(shù)值可以使用long long類型的reply-> integer字段來訪問
  • REDIS_REPLY_NIL:
    表示返回空數(shù)據(jù)
  • REDIS_REPLY_STRING:
    表示返回一個字符串楣富≡涞啵可以使用reply-> str來訪問。該字符串的長度可以使用reply->len來訪問纹蝴。
  • REDIS_REPLY_ARRAY:
    表示返回一個數(shù)組庄萎。例如mget,元素?cái)?shù)量存儲在reply->elements元素中塘安。每個元素也是一個redisReply對象糠涛,可以通過reply-> element [index]進(jìn)行訪問。

使用示例

//set
string set_ = "set China Beijing";
redisReply *reply = (redisReply *)redisCommand(redis_handle, set_.c_str());
if (reply!=NULL && reply->type==REDIS_REPLY_STATUS) {
    printf("%s\n", reply->str);
}
freeReplyObject(reply);

//mset
string mset_ = "set China Beijing Japan Tokyo";
redisReply *reply = (redisReply *)redisCommand(redis_handle, mset_.c_str());
if (reply!=NULL && reply->type==REDIS_REPLY_STATUS) {
    printf("%s\n", reply->str);
}
freeReplyObject(reply);

//get
string get_ = "get China";
redisReply *reply = (redisReply *)redisCommand(redis_handle, get_.c_str());
if (reply!=NULL && reply->type==REDIS_REPLY_STRING) {
    printf("%s\n", reply->str);
}
freeReplyObject(reply);  //使用reply以后必須要進(jìn)行釋放

//mget
string mget_ = "mget China Japan";
redisReply *reply = (redisReply *)redisCommand(redis_handle, mget_.c_str());
if (reply!=NULL && reply->type==REDIS_REPLY_ARRAY) {
    for (int i=0; i<reply->elements; ++i) {
        printf("%s\n", reply->element[i]->str);
    }
}
freeReplyObject(reply);

當(dāng)命令執(zhí)行發(fā)生錯誤時(shí)兼犯,reply為NULL忍捡,redis_handle中的err字段將被設(shè)置。一旦錯誤返回切黔,redis_handle不能被重用砸脊,你應(yīng)該建立一個新的連接。

redisReply *reply = (redisReply *)redisCommand(redis_handle, mget_str.c_str());
if (reply == NULL) {
    printf("Error:%s\n", redis_handle->errstr);
    redisReconnect(redis_handle);
}

進(jìn)階

由于hiRedis是非線程安全的纬霞,即一個redis_handle不能同時(shí)被多個線程使用脓规,因此通過連接池解決并發(fā)訪問redis的問題

#include <string>
#include <mutex>
#include "hiredis.h"

class redis_pool {

        public:
                redis_pool() {
                }

                ~redis_pool() {
                        for (int i=0; i<conn_num; ++i) {
                                if (conn_pool[i] != NULL) {
                                        redisFree(conn_pool[i]);
                                        conn_pool[i] = NULL;
                                }

                        }
                        delete [] conn_pool;

                        if (conn_flag != NULL) {
                                delete [] conn_flag;
                                conn_flag = NULL;
                        }
                }

                int init(std::string ip_, int port_, int conn_num_) {
                        ip = ip_;
                        port = port_;
                        conn_num = conn_num_;


                        conn_pool = new redisContext * [conn_num];
                        if (conn_pool == NULL) {
                                return 1;
                        }

                        conn_flag = new int[conn_num];
                        if (conn_flag == NULL) {
                                return 2;
                        }

                        for (int i=0; i<conn_num; ++i) {
                                conn_pool[i] = redisConnect(ip.c_str(), port);
                                if (conn_pool[i]==NULL || conn_pool[i]->err) {
                                        return 3;
                                }

                                conn_flag[i] = 0;
                        }

                        empty_num = conn_num;
                        current_conn = 0;

                        return 0;
                }

                redisContext *get_conn(int &id) {
                        if (empty_num == 0) {
                                return NULL;
                        }

                        mtx.lock();

                        while (conn_flag[current_conn] != 0) {current_conn = (current_conn+1) % conn_num;}

                        conn_flag[current_conn] = 1;
                        --empty_num;
                        id = current_conn;
                        current_conn = (current_conn+1) % conn_num;

                        mtx.unlock();

                        return conn_pool[id];
                }

                void put_conn(int id) {
                        if (id<conn_num && id>=0) {
                                mtx.lock();

                                conn_flag[id] = 0;
                                ++empty_num;

                                mtx.unlock();
                        }

                        return;
                }

        private:
                std::string ip;
                int port;
                int conn_num;

                redisContext **conn_pool;
                int *conn_flag;
                int empty_num;
                int current_conn;

                std::mutex mtx;

};

備注

  • redis對于空閑連接,會在一定時(shí)間內(nèi)關(guān)閉险领,當(dāng)hiRedis的連接被關(guān)閉時(shí)侨舆,通過redisCommand返回的reply為NULL秒紧,需要調(diào)用redisReconnect重連
  • 調(diào)用redisEnableKeepAlive并未達(dá)到保持連接的效果
  • 當(dāng)key不存在時(shí),redisCommand并不會返回失敗挨下,只是str成員會為NULL

參考

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末熔恢,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子臭笆,更是在濱河造成了極大的恐慌叙淌,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,542評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件愁铺,死亡現(xiàn)場離奇詭異鹰霍,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)茵乱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,596評論 3 385
  • 文/潘曉璐 我一進(jìn)店門茂洒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人瓶竭,你說我怎么就攤上這事督勺。” “怎么了斤贰?”我有些...
    開封第一講書人閱讀 158,021評論 0 348
  • 文/不壞的土叔 我叫張陵智哀,是天一觀的道長。 經(jīng)常有香客問我荧恍,道長瓷叫,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,682評論 1 284
  • 正文 為了忘掉前任送巡,我火速辦了婚禮赞辩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘授艰。我一直安慰自己,他們只是感情好世落,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,792評論 6 386
  • 文/花漫 我一把揭開白布淮腾。 她就那樣靜靜地躺著,像睡著了一般屉佳。 火紅的嫁衣襯著肌膚如雪谷朝。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,985評論 1 291
  • 那天武花,我揣著相機(jī)與錄音圆凰,去河邊找鬼。 笑死体箕,一個胖子當(dāng)著我的面吹牛专钉,可吹牛的內(nèi)容都是我干的挑童。 我是一名探鬼主播,決...
    沈念sama閱讀 39,107評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼跃须,長吁一口氣:“原來是場噩夢啊……” “哼站叼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起菇民,我...
    開封第一講書人閱讀 37,845評論 0 268
  • 序言:老撾萬榮一對情侶失蹤尽楔,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后第练,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體阔馋,經(jīng)...
    沈念sama閱讀 44,299評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,612評論 2 327
  • 正文 我和宋清朗相戀三年娇掏,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了呕寝。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,747評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡驹碍,死狀恐怖壁涎,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情志秃,我是刑警寧澤怔球,帶...
    沈念sama閱讀 34,441評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站浮还,受9級特大地震影響竟坛,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜钧舌,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,072評論 3 317
  • 文/蒙蒙 一担汤、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧洼冻,春花似錦崭歧、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,828評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至屋彪,卻和暖如春所宰,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背畜挥。 一陣腳步聲響...
    開封第一講書人閱讀 32,069評論 1 267
  • 我被黑心中介騙來泰國打工仔粥, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 46,545評論 2 362
  • 正文 我出身青樓躯泰,卻偏偏與公主長得像谭羔,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子斟冕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,658評論 2 350

推薦閱讀更多精彩內(nèi)容