初学Redis(2)——用Redis作为Mysql数据库的缓存
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了初学Redis(2)——用Redis作为Mysql数据库的缓存,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含4959字,纯文字阅读大概需要8分钟。
内容图文
用Redis作Mysql数据库缓存,必须解决2个问题。首先,应该确定用何种数据结构存储来自Mysql的数据;在确定数据结构之后,还要考虑用什么标识作为该数据结构的键。
直观上看,Mysql中的数据都是按表存储的;更微观地看,这些表都是按行存储的。每执行一次select查询,Mysql都会返回一个结果集,这个结果集由若干行组成。所以,一个自然而然的想法就是在Redis中找到一种对应于Mysql行的数据结构。Redis中提供了五种基本数据结构,即字符串(string)、列表(list)、哈希(hash)、集合(set)和有序集合(sorted set)。经过调研,发现适合存储行的数据结构有两种,即string和hash。
要把Mysql的行数据存入string,首先需要对行数据进行格式化。事实上,结果集的每一行都可以看做若干由字段名和其对应值组成的键值对集合。这种键值对结构很容易让我们想起Json格式。因此,这里选用Json格式作为结果集每一行的格式化模板。根据这一想法,我们可以实现将结果集格式化为若干Json对象,并将Json对象转化为字符串存入Redis的代码:
// 该函数把结果集中的每一行转换为一个Json格式的字符串并存入Redis的STRING结构中, // STRING键应该包含结果集标识符和STRING编号,形式如“cache.string:123456:1” string Cache2String(sql::Connection *mysql_connection, redisContext *redis_connection, sql::ResultSet *resultset, const string &resultset_id, int ttl) { if (resultset->rowsCount() == 0) { throw runtime_error("FAILURE - no rows"); } // STRING键的前缀,包含了结果集的标识符 string prefix("cache.string:" + resultset_id + ":"); unsigned int num_row = 1; // STRING编号,附加于STRING键的末尾,从1开始 sql::ResultSetMetaData *meta = resultset->getMetaData(); unsigned int num_col = meta->getColumnCount(); // 将结果集中所有行对应的所有STRING键存入该SET,SET键包含了结果集的标识符 string redis_row_set_key("resultset.string:" + resultset_id); redisReply *reply; string ttlstr; stringstream ttlstream; ttlstream << ttl; ttlstr = ttlstream.str(); resultset->beforeFirst(); // 将结果集中的每一行转为Json格式的字符串,将这些Json字符串存入STRING, // 每个STRING对应结果集中的一行 while (resultset->next()) { string redis_row_key; // STRING键名,由前缀和STRING编号组成 stringstream keystream; keystream << prefix << num_row; redis_row_key = keystream.str(); Json::Value row; for (int i = 1; i <= num_col; ++i) { string col_label = meta->getColumnLabel(i); string col_value = resultset->getString(col_label); row[col_label] = col_value; } Json::FastWriter writer; string redis_row_value = writer.write(row); // 将STRING键及Json格式的对应值对存入Redis reply = static_cast<redisReply*>(redisCommand(redis_connection, "SET %s %s", redis_row_key.c_str(), redis_row_value.c_str())); freeReplyObject(reply); // 将STRING键加入SET中 reply = static_cast<redisReply*>(redisCommand(redis_connection, "SADD %s %s", redis_row_set_key.c_str(), redis_row_key.c_str())); freeReplyObject(reply); // 设置STRING的过期时间 reply = static_cast<redisReply*>(redisCommand(redis_connection, "EXPIRE %s %s", redis_row_key.c_str(), ttlstr.c_str())); freeReplyObject(reply); ++num_row; } // 设置SET的过期时间 reply = static_cast<redisReply*>(redisCommand(redis_connection, "EXPIRE %s %s", redis_row_set_key.c_str(), ttlstr.c_str())); freeReplyObject(reply); return redis_row_set_key; // 返回SET键,以便于其他函数获取该SET中的内容 }
要把Mysql的行数据存入hash,过程要比把数据存入string直观很多。这是由hash的结构性质决定的——hash本身就是一个键值对集合:一个“父键”下面包含了很多“子键”,每个“子键”都对应一个值。根据前面的分析可知,结果集中的每一行实际上也是键值对集合。用Redis键值对集合表示Mysql键值对集合应该再合适不过了:对于结果集中的某一行,字段对应于hash的“子键”,字段对应的值就是hash“子键”对应的值,即结果集的一行刚好对应一个hash。这一想法的实现代码如下:
// 该函数把结果集中的每一行都存入一个HASH结构。HASH键应当包括结果集标识符和HASH编号, // 形如“cache.string:123456:1” string Cache2Hash(sql::Connection *mysql_connection, redisContext *redis_connection, sql::ResultSet *resultset, const string &resultset_id, int ttl) { if (resultset->rowsCount() == 0) { throw runtime_error("FAILURE - no rows"); } // HASH键的前缀,包含了结果集的标识符 string prefix("cache.hash:" + resultset_id + ":"); unsigned int num_row = 1; // HASH编号,附加于HASH键的末尾,从1开始 sql::ResultSetMetaData *meta = resultset->getMetaData(); unsigned int num_col = meta->getColumnCount(); // 将结果集中所有行对应的所有HASH键存入该SET,SET键包含了结果集的标识符 string redis_row_set_key("resultset.hash:" + resultset_id); redisReply *reply; string ttlstr; stringstream ttlstream; ttlstream << ttl; ttlstr = ttlstream.str(); // 结果集中的每一行对应于一个HASH,将结果集的所有行都存入相应HASH中 resultset->beforeFirst(); while (resultset->next()) { string redis_row_key; // HASH键名,由前缀和HASH编号组成 stringstream keystream; keystream << prefix << num_row; redis_row_key = keystream.str(); for (int i = 1; i <= num_col; ++i) { string col_label = meta->getColumnLabel(i); string col_value = resultset->getString(col_label); // 将结果集中一行的字段名和对应值存入HASH reply = static_cast<redisReply*>(redisCommand(redis_connection, "HSET %s %s %s", redis_row_key.c_str(), col_label.c_str(), col_value.c_str())); freeReplyObject(reply); } // 将HASH键加入SET中 reply = static_cast<redisReply*>(redisCommand(redis_connection, "SADD %s %s", redis_row_set_key.c_str(), redis_row_key.c_str())); freeReplyObject(reply); // 设置HASH的过期时间 reply = static_cast<redisReply*>(redisCommand(redis_connection, "EXPIRE %s %s", redis_row_key.c_str(), ttlstr.c_str())); freeReplyObject(reply); ++num_row; } // 设置SET的过期时间 reply = static_cast<redisReply*>(redisCommand(redis_connection, "EXPIRE %s %s", redis_row_set_key.c_str(), ttlstr.c_str())); freeReplyObject(reply); return redis_row_set_key; // 返回SET键,以便于其他函数获取该SET中的内容 }
至此,我们已经给出了两种存储Mysql结果集的方案,这就是我们在篇首提出的第一个问题,即选择何种数据结构存储Mysql结果集的答案。下一篇文章将研究第二个问题,即数据结构键的标识符选择问题。
原文:http://blog.csdn.net/qtyl1988/article/details/39519951
内容总结
以上是互联网集市为您收集整理的初学Redis(2)——用Redis作为Mysql数据库的缓存全部内容,希望文章能够帮你解决初学Redis(2)——用Redis作为Mysql数据库的缓存所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。