【7】cpp_redis hash容器 增删改查(避免死锁的解决方法) 【最后更新于2018-05-05】

本文介绍了一个使用 C++ Redis 客户端时可能导致死锁的问题,并提供了解决方案。通过每次 get 或 set 操作后立即提交 commit 来避免死锁。
部署运行你感兴趣的模型镜像

直接上代码:该代码在循环中反复执行可能会导致死锁,解决的办法见后文

#include <cpp_redis/cpp_redis>
#include <iostream>
#include "TestRedis.h"
using namespace std;

#ifdef _WIN32
#include <Winsock2.h>
#endif /* _WIN32 */
#pragma comment( lib, "ws2_32.lib")//最好的方法是包含在项目属性中,因为那样可以根据Debug、Release、x86、x64来区分。这里仅仅是为了突出引用了这个库写在这里 

bool StartWSA(void)
{
	//! Windows netword DLL init
	WORD version = MAKEWORD(2, 2);
	WSADATA data;

	if (WSAStartup(version, &data) != 0)
	{
		std::cerr << "WSAStartup() failure" << std::endl;
		return false;
	}
	return true;
}
void StopWSA(void)
{
	WSACleanup();
}

typedef std::function<void(cpp_redis::reply&)> reply_callback_t;
void fun_call_back(cpp_redis::reply& _reply)
{
	std::cout << _reply << endl;
}
void cpp_redis_connect(cpp_redis::redis_client& _client)
{
	//! Enable logging
	cpp_redis::active_logger = std::unique_ptr<cpp_redis::logger>(new cpp_redis::logger);
	_client.connect("127.0.0.1", 6379, [](cpp_redis::redis_client&)
	{
		std::cout << "client disconnected (disconnection handler)" << std::endl;
	});
}
int main(void)
{
	//启动Windows网络通信库
	if (!StartWSA())
	{
		return -1;
	}
	try
	{
		cpp_redis::redis_client client;
		cpp_redis_connect(client);

		//hash 增删改查
		
		//新增:批量
		vector<pair<string, string>> vecPair = { { "key1", "value1" },{ "key2", "value2" },{ "key3", "value3" } };
		client.hmset("MyHashSet", vecPair, [](cpp_redis::reply& _reply) {
			std::cout<<"新增批量:key1 key2 key3. " << _reply << endl;
		});
		//新增:单个
		client.hset("MyHashSet", "key4", "value4", [](cpp_redis::reply& _reply) {
			std::cout << "新增单个:key4. " << _reply << endl;
		});
		
		//查询:批量
		client.hgetall("MyHashSet", [](cpp_redis::reply& _reply)//这种查询适合批量返回结果的情形,避免了来回的网络传输
		{
			std::cout << "查询批量:MyHashSet" << endl;
			for each (auto& var in _reply.as_array())//这种方式返回的key 和value是挨着放的,因为是getall,所以不存在nill的情况(只要返回的vector的size>0)
			{
				std::cout << var << endl;
			}
		});
		
		//修改:单个
		client.hset("MyHashSet", "key4", "VALUE4", [](cpp_redis::reply& _reply) {
			std::cout << "修改单个:MyHashSet key4 VALUE4. " << _reply << endl;
		});
		//查询:单个
		client.hget("MyHashSet", "key4", [](cpp_redis::reply& _reply)
		{
			std::cout << "查询单个:MyHashSet key4. " << _reply << endl;
		});
		
		//删除:批量
		vector<string> vecKeys = { "key1", "key2", "key_not_exist" };//删除不存在的key不会有任何影响,对客户端来说
		client.hdel("MyHashSet", vecKeys, [](cpp_redis::reply& _reply) {
			std::cout << "删除:批量:MyHashSet : key1, key2, key_not_exist. " << _reply << endl;
		});
		
		//查询:批量
		client.hgetall("MyHashSet", [](cpp_redis::reply& _reply)//这种查询适合批量返回结果的情形,避免了来回的网络传输
		{
			std::cout << "查询:批量:MyHashSet :" <<endl;
			for each (auto& var in _reply.as_array())//这种方式返回的key 和value是挨着放的,因为是getall,所以不存在nill的情况(只要返回的vector的size>0)
			{
				std::cout << var << endl;
			}
		});

		// synchronous commit 同步提交:将上面的所有get set 等方法依次执行完, no timeout
		client.sync_commit();
	}
	catch (cpp_redis::redis_error e)
	{
		std::cout << e.what() << endl;
	}

	//关闭Windows网络通信库
	StopWSA();

	return 0;
}

输出结果:符合预期


上面的代码封装成test_fun_redis_hash()的话,下面的代码会死锁:

int main(int argc, char* argv[])
{
	//启动Windows网络通信库
	if (!StartWSA())
	{
		return -1;
	}

	while (true)
	{
		test_use_cpp_redis_with_rapidjson();
		Sleep(10);
	}


	//关闭Windows网络通信库
	StopWSA();

	return 0;
}

主要原因是:

1 cpp_redis采用延迟执行的方式,get set 都是把数据追加到client的成员buffer里,等commit的时候一起执行。

2 当连续执行多个get的时候,可以最后一次性commit

3 当连续执行多个set的时候,可以最后一次性commit

4 当多个get/set针对同一个redis库中的对象交叉执行时(可能会造成死锁),避免死锁的办法就是每次get/set之后就提交一次commit

上文问题解决办法:

1 在每一个get/set之后都commit一次(连续多个get可以用一个commit,连续多个set也可以用一个commit)

修改后不会死锁的代码如下:

#include <cpp_redis/cpp_redis>
#include <iostream>
#include "TestRedis.h"
using namespace std;

#ifdef _WIN32
#include <Winsock2.h>
#endif /* _WIN32 */
#pragma comment( lib, "ws2_32.lib")//最好的方法是包含在项目属性中,因为那样可以根据Debug、Release、x86、x64来区分。这里仅仅是为了突出引用了这个库写在这里 

bool StartWSA(void)
{
	//! Windows netword DLL init
	WORD version = MAKEWORD(2, 2);
	WSADATA data;

	if (WSAStartup(version, &data) != 0)
	{
		std::cerr << "WSAStartup() failure" << std::endl;
		return false;
	}
	return true;
}
void StopWSA(void)
{
	WSACleanup();
}

typedef std::function<void(cpp_redis::reply&)> reply_callback_t;
void fun_call_back(cpp_redis::reply& _reply)
{
	std::cout << _reply << endl;
}
void cpp_redis_connect(cpp_redis::redis_client& _client)
{
	//! Enable logging
	cpp_redis::active_logger = std::unique_ptr<cpp_redis::logger>(new cpp_redis::logger);
	_client.connect("127.0.0.1", 6379, [](cpp_redis::redis_client&)
	{
		std::cout << "client disconnected (disconnection handler)" << std::endl;
	});
}

void test_use_cpp_redis_with_rapidjson(void)
{
	try
	{
		cpp_redis::redis_client client;
		cpp_redis_connect(client);

		string val;

		//hash 增删改查

		//新增:批量
		vector<pair<string, string>> vecPair = { { "key1", "value1" },{ "key2", "value2" },{ "key3", "value3" } };
		client.hmset("MyHashSet", vecPair, [&](cpp_redis::reply& _reply) {
			std::cout << "新增批量:key1 key2 key3. " << _reply << endl;
		});
		client.sync_commit();
		//新增:单个
		client.hset("MyHashSet", "key4", "value4", [&](cpp_redis::reply& _reply) {
			std::cout << "新增单个:key4. " << _reply << endl;
		});
		client.sync_commit();

		//查询:批量
		client.hgetall("MyHashSet", [&](cpp_redis::reply& _reply)//这种查询适合批量返回结果的情形,避免了来回的网络传输
		{
			std::cout << "查询批量:MyHashSet" << endl;
			for each (auto& var in _reply.as_array())//这种方式返回的key 和value是挨着放的,因为是getall,所以不存在nill的情况(只要返回的vector的size>0)
			{
				std::cout << var.as_string() << endl;
			}
		});
		client.sync_commit();

		//修改:单个
		client.hset("MyHashSet", "key4", "VALUE4", [&](cpp_redis::reply& _reply) {
			std::cout << "修改单个:MyHashSet key4 VALUE4. " << _reply << endl;
		});
		client.sync_commit();
		//查询:单个
		client.hget("MyHashSet", "key4", [&](cpp_redis::reply& _reply)
		{
			std::cout << "查询单个:MyHashSet key4. " << _reply.as_string() << endl;
			val = _reply.as_string();
			cout << "val:" << val << endl;
		});
		client.sync_commit();

		//删除:批量
		vector<string> vecKeys = { "key1", "key2", "key_not_exist" };//删除不存在的key不会有任何影响,对客户端来说
		client.hdel("MyHashSet", vecKeys, [&](cpp_redis::reply& _reply) {
			std::cout << "删除:批量:MyHashSet : key1, key2, key_not_exist. " << _reply << endl;
		});
		client.sync_commit();

		//查询:批量
		client.hgetall("MyHashSet", [&](cpp_redis::reply& _reply)//这种查询适合批量返回结果的情形,避免了来回的网络传输
		{
			std::cout << "查询:批量:MyHashSet :" << endl;
			for each (auto& var in _reply.as_array())//这种方式返回的key 和value是挨着放的,因为是getall,所以不存在nill的情况(只要返回的vector的size>0)
			{
				std::cout << var.as_string() << endl;
			}
		});

		// synchronous commit 同步提交:将上面的所有get set 等方法依次执行完, no timeout
		client.sync_commit();
	}
	catch (cpp_redis::redis_error e)
	{
		std::cout << e.what() << endl;
	}
}

int main(void)
{
	//启动Windows网络通信库
	if (!StartWSA())
	{
		return -1;
	}
	while (true)
	{
		test_use_cpp_redis_with_rapidjson();
		Sleep(10);
	}

	//关闭Windows网络通信库
	StopWSA();

	return 0;
}


您可能感兴趣的与本文相关的镜像

ACE-Step

ACE-Step

音乐合成
ACE-Step

ACE-Step是由中国团队阶跃星辰(StepFun)与ACE Studio联手打造的开源音乐生成模型。 它拥有3.5B参数量,支持快速高质量生成、强可控性和易于拓展的特点。 最厉害的是,它可以生成多种语言的歌曲,包括但不限于中文、英文、日文等19种语言

1. TCP/UDP通信流程相关问题 可能提问: TCP和UDP的区别是什么? TCP三次握手和四次挥手的过程是怎样的?为什么是三次握手? 如何在C++中实现一个TCP服务器和客户端? UDP通信是否可靠?如何保证数据包的有序和完整? socket编程中常用的函数有哪些?例如bind(), listen(), accept()的作用? 2. 多线程编程相关问题 可能提问: std::thread如何创建和管理线程?线程如何传参? 多线程中如何避免资源竞争?什么是死锁?如何避免? std::mutex和std::lock_guard的使用场景是什么? std::map在多线程下是否线程安全?如何保证线程安全? 是否了解std::atomic和CAS操作? 3. STL容器与C++语法相关问题 可能提问: std::map和std::unordered_map的区别是什么?各自适用场景? std::map插入、查找的时间复杂度是多少? C++11新特性中你常用哪些?如std::shared_ptr、std::move、lambda表达式? 智能指针了解吗?unique_ptr和shared_ptr的区别? 4. 系统设计与调试能力相关问题 可能提问: 后端模块如何处理并发连接?线程池了解吗?你有没有实现过? 项目中是如何管理用户和房间信息的?是否考虑过性能瓶颈? 遇到过哪些多线程下的bug?你是如何调试并解决的? 如何保证代码的可维护性和规范性?是否使用过日志系统? 5. 项目细节与个人贡献相关问题 可能提问: 你在项目中具体负责哪些模块?遇到了哪些挑战? 有没有参与系统性能优化?比如高并发下如何优化线程调度? 如果让你重新设计这个项目,你会改进哪些方面? 知识点 TCP/UDP通信流程:掌握网络通信的基本原理和编程接口。 多线程编程:理解线程控制、同步机制和资源共享问题。 STL容器使用:熟练使用标准库容器进行数据管理和性能优化。 给出答案
09-25
评论 3
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

C++程序员Carea

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值