asynDBCenter
asynDBCenter是GS和DBCenter之间的模块,有了他GS访问数据库就是异步的了,以前是同步的,加入某个操作很耗时那么GS就在那等待这个返回值。
.对于std::queue哪些情况要加锁,哪些不加
push、pop操作100%必须加锁,front和back操作是只读的,不会破坏状态,但是有读写顺序要求时还是要同步。
一般empty和size就不要锁了,对于锁 里面有两个容器
std::queue<CmdPkt> m_queueCmdPkt;//命令包容器
std::queue<HrPkt> m_queueHrPkt;//相应包容器 .m_queueCmdPkt这个容器是什么时候放入的呢?
玩家登录获取userid的函数
/*
std::condition_variable
进入时获得锁,wait时释放,当别人通知的时候重新获得
锁的状态在线程阻塞之前和阻塞之后相同
*/
void asynDBCenter::push_cmd(CmdPkt pkt)
{
{
std::unique_lock<std::mutex> cond_lock(m_cond_mtx);//这里加锁,主要还是同步,但没有多大意义,因为没有对公共的变量操作,不是很确定
{
std::unique_lock<std::mutex> cmd_lock(m_cmd_mtx);
m_queueCmdPkt.push(pkt);
}
}
m_cond.notify_one();//取的线程阻塞了,现在通知
} void asynDBCenter::getUserID(std::function<void(int hr)>* on_getUserID, char* userName)
{
boost::property_tree::ptree pt;
pt.put("userName", userName);
std::stringstream os;
boost::property_tree::write_json(os, pt); CmdPkt pkt;
pkt.json = os.str();
pkt.process = std::bind(&asynDBCenter::isGetUserID, this, ph::_1, ph::_2);
pkt.on_process = (void*)on_getUserID;
push_cmd(pkt);//此时放入
} .什么时候从m_queueCmdPkt容器获取
asynDBCenter::asynDBCenter()
{
m_dbcenter.reset(NEW(DBCenter));
m_spThread.reset(new std::thread(std::bind(&asynDBCenter::threadLoop, this)));
} bool asynDBCenter::get_front_and_pop(CmdPkt& pkt)
{
//对于这句不放到锁里面,我在想如果执行这句不为空,然后执行到下面的时候别的线程把他变成空了,岂不是错了
//但我看也就只有这一个对方在取,所以不放到锁里面也是可以的
if(m_queueCmdPkt.empty())
return false;
{
std::unique_lock<std::mutex> lock(m_cmd_mtx);
pkt = m_queueCmdPkt.front();
m_queueCmdPkt.pop();
}
return true;
} void asynDBCenter::threadLoop()
{
while()
{
CmdPkt pkt;
if(get_front_and_pop(pkt))
{
pkt.process(pkt.on_process, pkt.json.c_str());//访问数据库
continue;
}
{//表示GS发过来的命令没有了
std::unique_lock<std::mutex> cond_lock(m_cond_mtx);
m_cond.wait(cond_lock);//使用unique_lock锁住当前线程,然后另一个线程在相同的m_cond上调用notify时此线程才会解除阻塞
}
}
} .访问数据库 void asynDBCenter::push_hr(HrPkt pkt)
{
std::unique_lock<std::mutex> hr_lock(m_hr_mtx);
m_queueHrPkt.push(pkt);//直接放入
} void asynDBCenter::isGetUserID(void* on_process, const char* arg)
{
std::stringstream ros(arg);
boost::property_tree::ptree pt;
boost::property_tree::read_json(ros, pt);
std::string userName = pt.get<std::string>("userName"); int ret = m_dbcenter->getUserID(const_cast<char*>(userName.c_str())); boost::property_tree::ptree ret_pt;//将数据库返回的值封装成json
ret_pt.put("userID", ret);
std::stringstream wos;
boost::property_tree::write_json(wos, ret_pt);
//放到结果包里面去
HrPkt pkt;
pkt.hr = wos.str();
pkt.on_process = on_process;
pkt.process = std::bind(&asynDBCenter::onGetUserID, this, ph::_1, ph::_2);
push_hr(pkt);
} .函数回调
void asynDBCenter::drive()
{ HrPkt pkt;
for (;;)
{
if(!m_queueHrPkt.size())//drive一下就把m_queueHrPkt所有的命令都执行了
break ;
{
std::unique_lock<std::mutex> hr_lock(m_hr_mtx);
pkt = m_queueHrPkt.front();
m_queueHrPkt.pop();
}
pkt.process(pkt.on_process, pkt.hr.c_str(), pkt.len);
}
} for (;;)
{
procMapSendData();//test1 m_asyndb->drive();//这个是在gs里面drive的 bool hr = process_loop(pkt);
procMapSendData();
//......
}
GS线程发命令到asyn,asyn线程去访问数据库,然后GS线程去驱动,然后回调GS里面的函数
function在这里面起了很大的作用

asynDBcenter(复习)的更多相关文章

  1. iOS总结_UI层自我复习总结

    UI层复习笔记 在main文件中,UIApplicationMain函数一共做了三件事 根据第三个参数创建了一个应用程序对象 默认写nil,即创建的是UIApplication类型的对象,此对象看成是 ...

  2. vuex复习方案

    这次复习vuex,发现官方vuex2.0的文档写得太简略了,有些看不懂了.然后看了看1.0的文档,感觉很不错.那以后需要复习的话,还是先看1.0的文档吧.

  3. 我的操作系统复习——I/O控制和系统调用

    上篇博客介绍了存储器管理的相关知识——我的操作系统复习——存储器管理,本篇讲设备管理中的I/O控制方式和操作系统中的系统调用. 一.I/O控制方式 I/O就是输入输出,I/O设备指的是输入输出设备和存 ...

  4. 复习(1)【Maven】

    终于开始复习旧知识了,有输入必然要有输出.输入和输出之间的内化过程尤为重要,在复习的同时,真正把学到的东西积淀下来,加深理解. Maven项目概念与配置 Maven是一个项目管理和综合工具.Maven ...

  5. 《CSS权威指南》基础复习+查漏补缺

    前几天被朋友问到几个CSS问题,讲道理么,接触CSS是从大一开始的,也算有3年半了,总是觉得自己对css算是熟悉的了.然而还是被几个问题弄的"一脸懵逼"... 然后又是刚入职新公司 ...

  6. JS复习--更新结束

    js复习-01---03 一 JS简介 1,文档对象模型 2,浏览器对象模型 二 在HTML中使用JS 1,在html中使用<script></script>标签 2,引入外部 ...

  7. jQuery 复习

    jQuery 复习 基础知识 1, window.onload $(function(){});   $(document).ready(function(){}); 只执行函数体重的最后一个方法,事 ...

  8. jQuery5~7章笔记 和 1~3章的复习笔记

    JQery-05 对表单和表格的操作及其的应用 JQery-06 jQuery和ajax的应用 JQery-07 jQuery插件的使用和写法 JQery-01-03 复习 之前手写的笔记.实在懒得再 ...

  9. HTML和CSS的复习总结

    HTML(Hypertext Markup Language)超文本标记语言:其核心就是各种标记!<html> HTML页面中的所有内容,都在该标签之内:它主要含<head>和 ...

随机推荐

  1. (转)Linux下Oracle启动、建立表空间、用户、授权、数据库导入导出

    Linux系列 启动1.启动数据库实例,分为两步:第一步,启动监听:第二步,启动数据库实例. 1.1进入到sqlplus启动实例 [oracle@redhat ~]$ su - oracle      ...

  2. HTTP状态码参考

    1. HTTP状态码意义 客户机与服务器建立连接后,发送一个请求给服务器(如:Get /index.html http/1.1),在服务器接到请求后,给予客户机相应的响应信息,包括该信息的协议版本号. ...

  3. asp.net visio com接口 asp.net和visio混合编程

    主要介绍asp.net调用visio com的基本用法,主要用于控制visio图形背景色,文字显示等. 主要步骤: 1. 项目中引用COM组件,找到Mircosoft Visio 14 Type Li ...

  4. id,class,name区别

    id,class,name区别 id:标签唯一标识,好比我们身份证号码,具有唯一性.JS常用document,getGlementBy(id). class:标签的类别,可重复使用,CSS常用. na ...

  5. Tables for condition techniques

    T682i  -- Access sequence and the tables   T685  -- Condition types and Access sequences   T683s  -- ...

  6. 用canvas 绘制的饼状统计图、柱状统计图、折线统计图

    canvas 绘制的饼状统计图 canvas 绘制的柱状统计图 canvas 绘制的折线统计图

  7. 批处理测试局域网网络连通性ping1-255

    for /l %%1 in (1 1 255)do ping /n 1 192.168.1.%%1       ##bat下 运行 for /l %i in (1,1,254) do ping -n ...

  8. mac OS X下安装Redis及Thinkphp3.1使用Redis

    一.安装Redis 1.安装Homebrew 在终端输入ruby -e "$(curl -fsSL https://raw.github.com/Homebrew/install/maste ...

  9. Apache 2.4 多站点配置记录

    基于域名的虚拟主机配置 <VirtualHost *:80> DocumentRoot /var/www/cms ServerName www.example.com RewriteEng ...

  10. EcilpsePHP studio 3.0 运行(run)环境配置

    EcilpsePHP studio 3.0的界面与 MyEclipse操作界面基本一样,熟悉后者的对于EcilpsePHP studio 的使用学习就不会太难了. 安装好EPP后,新建项目--> ...