项目主要实现的功能:注册,登陆,获取好友列表,一对一聊天,消息群发,离线消息缓存,下线。
需要的工具:libevent,json,mysql,memcached
开发平台:vmware
基本项目思路:
客户端:利用多线程实现,采取输入命令的方式让用户选择服务类型,根据不同的服务类型调用不同的处理函数,完成请求,利用json和自定义的上层协议完成和服务器端的数据的交互。在登陆成功之后启动一个线程用于接受服务器端的消息。
服务器端:利用多线程并行(与并发有区别),结合半同步半异步的网络模型完成线程之间的任务分工,利用socket_pair完成线程之间的数据交互。
主线程负责接收用户的连接并将套接字按照子线程压力分发给子线程监听,子线程利用libevent实现I/O复用监听客户端的套接字,利用MVC模式通过判断客户端的请求类型调用不同的视图处理用户请求。
利用MySQL数据库存储用户相关信息以及离线消息,利用memcached实现数据库和内存之间的高速缓存。
项目基本框图:
在开始项目之前,让我们来单独实现简单的libevent,json,mysql.
libevent的客户端:#include
#include
#include
#include
#include
using namespace std;
int main()
{
int fd = socket(AF_INET,SOCK_STREAM,0);
if( -1 == fd)
{
cerr<
return 0;
}
struct sockaddr_in saddr;
saddr.sin_family = AF_INET;
saddr.sin_port = htons(6000);
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
if(-1 == connect(fd,(struct sockaddr*)&saddr,sizeof(saddr)))
{
cerr<
return 0;
}
//发送hello
int i = 0;
while(i < 2)
{
if(-1 == send(fd,"hello",6,0))
{
cerr<
return 0;
}
i++;
}
//接收 输出
char buff[100] = {0};
if(0 < recv(fd,buff,99,0))
{
cout<
sleep(100);
}
return 0;
}
libevent的服务器:
#include
#include
#include
#include
#include
using namespace std;
void cli_cb(int fd,short event,void *arg)//接受数据
{
struct event_base*lib_base = (struct event_base*)arg;
char buff[100] = {0};
if(0< recv(fd,buff,99,0))
{
cout<
if(-1 == send(fd,"ok",3,0))
{
cerr<
return;
}
}
else
{
close(fd);
//event_free(struct event *);
}
}
void listen_cb(int fd,short event,void* arg)//进行客户端服务器连接
{
struct event_base* lib_base = (struct event_base*)arg;
struct sockaddr_in caddr;
socklen_t len = sizeof(caddr);
int cli_fd = accept(fd,(struct sockaddr*)&caddr,&len);//三次握手
if(-1 == cli_fd)
{
cerr<
return;
}
//将clifd加入到事件列表
struct event* cli_event = event_new(lib_base,cli_fd,EV_READ|EV_PERSIST,cli_cb,lib_base);
if(NULL == cli_event)
{
cerr<
return;
}
event_add(cli_event,NULL);
}
int main()
{
struct event_base *lib_base = event_base_new();//事件库
int fd = socket(AF_INET,SOCK_STREAM,0);
if(-1 == fd)
{
cerr<
return 0;
}
struct sockaddr_in saddr;
saddr.sin_family = AF_INET;
saddr.sin_port = htons(6000);
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
if(-1 == bind(fd,(struct sockaddr*)&saddr,sizeof(saddr)))
{
cerr<
return 0;
}
if(-1 == listen(fd,20))//监听队列(20个)
{
cerr<
return 0;
}
//添加到libevent
//创建事件
struct event* listen_event = event_new(lib_base,fd,EV_READ|EV_PERSIST,listen_cb,lib_base);//申请一个监听事件 所有事件放base 触发回调函数
if(NULL == listen_event)
{
cerr<
return 0;
}
//将事件添加到事件列表
event_add(listen_event,NULL);
//循环监听
event_base_dispatch(lib_base);
return 0;
}
json的客户端:
#include
#include
#include
#include
#include
#include
using namespace std;
int main()
{
int fd = socket(AF_INET,SOCK_STREAM,0);
if(-1 == fd)
{
cerr<
return 0;
}
struct sockaddr_in saddr;
saddr.sin_family = AF_INET;
saddr.sin_port = htons(6000);
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
if(-1 == connect(fd,(struct sockaddr*)&saddr,sizeof(saddr)))
{
cerr<
return 0;
}
//封装json包,也可以用buff代替" "内容,在json里写入数据后发送
Json::Value val;
val["name"] = "zhangsan";
val["pw"] = "123456";
//发送json包
if(0 >= send(fd,val.toStyledString().c_str(),strlen(val.toStyledString().c_str()),0))
{
cerr<
return 0;
}
return 0;
}json的服务器:
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
int main()
{
int fd = socket(AF_INET,SOCK_STREAM,0);
if(-1 == fd)
{
cerr<
return 0;
}
struct sockaddr_in saddr;
saddr.sin_family = AF_INET;
saddr.sin_port = htons(6000);
saddr.sin_addr.s_addr = inet_addr("127.0.0.1");
if(-1 == bind(fd,(struct sockaddr *)&saddr,sizeof(saddr)))
{
cerr<
return 0;
}
if(-1 == listen(fd,20))
{
cerr<
return 0;
}
struct sockaddr_in caddr;
socklen_t len = sizeof(caddr);
int clifd = accept(fd,(struct sockaddr*)&caddr,&len);
if(-1 == clifd)
{
cerr<
return 0;
}
//接收客户端消息
char buff[100] = {0};
if(0 >= recv(clifd,buff,99,0))
{
cerr<
return 0;
}
cout<
Json::Value val;
Json::Reader read;
//解析json包
if(-1 == read.parse(buff,val))
{
cerr<
return 0;
}
//结果输出
cout<
cout<
return 0;
}
mysql:
#include
#include
#include
#include
using namespace std;
int main()
{
MYSQL *mpcon = mysql_init((MYSQL *)0);
MYSQL_RES *mp_res;
MYSQL_ROW mp_row;
//连接mysql
if(!mysql_real_connect(mpcon,"127.0.0.1","root","123456",NULL,3306,NULL,0))
{
cerr<
return 0;
}
//选择数据库
if(mysql_select_db(mpcon,"stu"))
{
cerr<
return 0;
}
//输入指令
char cmd[100] = "insert into user values('lisi','456789');";
if(mysql_real_query(mpcon,cmd,strlen(cmd)))
{
cerr<
return 0;
}
//输入指令
char cmd1[100] = "insert into user values('wangwu','456789');";
if(mysql_real_query(mpcon,cmd1,strlen(cmd1)))
{
cerr<
return 0;
}
//查找
char cmd2[100] = "select * from user where name='lisi';";
if(mysql_real_query(mpcon,cmd2,strlen(cmd2)))
{
cerr<
return 0;
}
//接收指令的返回值
mp_res = mysql_store_result(mpcon);
//接收返回值的一行
mp_row = mysql_fetch_row(mp_res);
cout<
//查找
char cmd3[100] = "select * from user;";
if(mysql_real_query(mpcon,cmd3,strlen(cmd3)))
{
cerr<
return 0;
}
//接收指令的返回值
mp_res = mysql_store_result(mpcon);
//接收返回值的一行
while(mp_row = mysql_fetch_row(mp_res))
{
cout<
}
return 0;
}