前言:

  最近在学进程间通信,所以做了一个小项目练习一下。主要用消息队列和socket(UDP)实现这个系统,并数据库存储数据,对C语言操作数据库不熟悉的可以参照我的这篇博客:https://www.cnblogs.com/liudw-0215/p/9593414.html,所有代码提交我的Github上,地址:https://github.com/ldw0215/Chat-System.git,可以自行下载,然后make一下就可以了。

   一、项目要求

  1. 要求实现用户注册、用户登录功能,密码需加密显示
  2. 要求实现聊天功能,双方能互发消息
  3. 数据要求数据库存储

  二、架构解析

  主要流程图如下:

    

  主要有客户端(用户)和服务端,客户端发送注册、登录请求,服务端回应请求,并且双方可以模拟聊天(互相发送消息),主要客户端请求界面见下图:

  

  注册、登录使用消息队列进行通信的,聊天是通过socket(UDP)实现的!数据存在数据库中,需要一张数据表,建表数据语句如下:

  

CREATE TABLE `user` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`name` varchar(64) NOT NULL DEFAULT '',
`password` varchar(64) NOT NULL DEFAULT '',
`check` varchar(64) NOT NULL DEFAULT '',
PRIMARY KEY (`id`),
UNIQUE KEY `name` (`name`)
) ENGINE=InnoDB AUTO_INCREMENT=52 DEFAULT CHARSET=utf8;

  三、客户端实现

  client.c创建不同的消息队列的键,根据不同的消息类型的进行发送,并等待服务端响应,client.c代码如下:

  

#include "my.h"

Msg m;
Msg_stoc msg_stoc; static int msgid_ctos;
static int msgid_stoc; void showmenu()
{
puts("-------CHAT----------");
puts("| 1:发送 2:接收 |");
puts("| 3:退出 |");
puts("--------------------");
} void show()
{
puts("-------CHAT----------");
puts("| 1:注册 2:登录 |");
puts("| 0:退出 |");
puts("--------------------");
} void send1()
{
printf("%s","send");
char buf[] = {'\0'};
char str[] = {'\0'};
struct sockaddr_in dui,zj;
int n;
short x;
int sockfd = socket(AF_INET,SOCK_DGRAM,);
if(sockfd < )
{
perror("socket");
exit(-); }
zj.sin_family = AF_INET;
zj.sin_port = htons();
zj.sin_addr.s_addr = htonl(INADDR_ANY);
n = bind(sockfd,(struct sockaddr *)&zj,sizeof(zj));
if(n < )
{
close(sockfd);
perror("bind");
exit(-); }
//puts("请输入对方号码 端口 IP ");
//scanf("%hd%s",&x,buf);
getchar();
dui.sin_addr.s_addr = inet_addr("10.10.3.129");
dui.sin_port = htons();
dui.sin_family = AF_INET;
puts("请输入想要发送的内容:");
//gets(str);
fgets(str,,stdin); n = sendto(sockfd,str,sizeof(str),,(struct sockaddr *)&dui,sizeof(dui));
if(n <= )
{
close(sockfd);
perror("sendto");
exit(-); }
close(sockfd);
return; }
#if 1
void asend(int sockfd,struct sockaddr_in dui)
{
char buf[] = {'\0'};
int n;
puts("请输入要回复的内容:");
fgets(buf,,stdin);
//gets(buf);
n = sendto(sockfd,buf,sizeof(buf),,(struct sockaddr*)&dui,sizeof(dui));
if(n <= )
{
perror("sendto");
close(sockfd);
exit(-); }
close(sockfd);
return ;
} void choose1(char ch ,int sockfd,struct sockaddr_in dui)
{
switch(ch)
{
case 'a':
asend(sockfd,dui);
break;
case 'n':
break;
default:
puts("input error!");
break; }
}
#endif
void recv1()
{
struct sockaddr_in dui,zj;
socklen_t len = sizeof(dui);
int n;
char buf[] = {'\0'};
char ch;
int sockfd = socket(AF_INET,SOCK_DGRAM,);
if(sockfd < )
{
perror("socket");
exit(-); }
zj.sin_family = AF_INET;
zj.sin_port = htons();
zj.sin_addr.s_addr = htonl(INADDR_ANY);
n = bind(sockfd,(struct sockaddr *)&zj,sizeof(zj));
if(n < )
{
close(sockfd);
perror("bind");
exit(-); }
n = recvfrom(sockfd,buf,sizeof(buf),,(struct sockaddr *)&dui,&len);
if(n <= )
{
close(sockfd);
perror("recvfrom");
exit(-);
}
puts(buf);
puts("是否要回复: 回复-》a,不回复-》n");
ch = getchar();
getchar();
choose1(ch,sockfd,dui);
} void choose(int ch)
{
printf("%d",ch);
switch(ch)
{
case :
m.type = ;
msgsnd(msgid_ctos,&m,sizeof(m)-sizeof(m.type),);
send1();
break;
case :
recv1();
break;
case :
exit();
break;
default:
puts("input error!");
break; }
} void regis()
{
printf("请输入姓名:");
scanf("%s",m.name); printf("请输入密码:");
scanf("%s",m.passwd);
m.type=;
} void login(void)
{
printf("请输入用户名:");
scanf("%s",&m.name); printf("请输入密码:");
scanf("%s",&m.passwd);
m.type=;
} int main(int argc ,char *argv[])
{
msgid_ctos=get_ctos_msg();
msgid_stoc=get_stoc_msg(); int temptype; while()
{
show();
int a = ;
scanf("%d",&a);
getchar();
switch(a)
{
case :return ;
case :regis();temptype=;break; //注册
case :login();temptype=;break; //登录
} int ret;
long type=; printf("name:%s,passwd:%s\n", m.name,m.passwd);
msgsnd(msgid_ctos,&m,sizeof(m)-sizeof(m.type),); sleep();
msgrcv(msgid_stoc,&msg_stoc,sizeof(msg_stoc),temptype,); if( == strcmp(msg_stoc.check,"yes"))
{
printf("%s",msg_stoc.info);
break;
}
} while()
{
showmenu();
int ch;
puts("请输入功能");
scanf("%d",&ch);
getchar();
printf("%d",ch);
choose(ch);
}
return ;
}

  注意,不同的通信,要用创建不同消息队列的键,并且消息类型也要不同!

  四、服务端实现

  服务端主要接送并响应客户端,主要创建不同的子进程,然后调用exec族函数,调用二进制文件,并通过消息队列接收阻塞执行,并建立信号,检测Ctrl+c信号,是进程退出,服务端响应截图如下:

  

  其实现代码如下:

  

#include"my.h"

static pid_t sub_pid[];

static int msgid_ctos;
static int msgid_stoc; void sigint(int signum)
{
int i;
for(i=;i<;i++)
{
kill(sub_pid[i],SIGKILL);
} } int main(int argc,char*argv[])
{
signal(SIGINT,sigint); msgid_ctos=get_ctos_msg();
msgid_stoc=get_stoc_msg(); sub_pid[]=vfork();
if(==sub_pid[])
{
execl("register","register",NULL);
} sub_pid[]=vfork();
if(==sub_pid[])
{
execl("login","login",NULL);
} sub_pid[]=vfork();
if(==sub_pid[])
{
execl("chat","chat",NULL);
} wait(NULL);
return ;
}

  四、各模块及数据库解析

  数据库是通过数据库函数实现的,需要头文件<mysql.h>,并链上动态库-I/usr/include/mysql -L/usr/lib64/mysql -lmysqlclient,数据量比较小,之后还要考虑优化的问题;

  注册、登录、聊天都是不同的.c文件生成二进制实现的:

  注册通过消息队列接收用户名和密码存入数据库,代码如下:

  

#include"my.h"

Msg per;
Msg_stoc msg_stoc; static int msgid_ctos;
static int msgid_stoc; void open_cli()
{
MYSQL conn;
int res;
//MYSQL_RES * result;
//MYSQL_ROW row;
mysql_init(&conn); //第三、四和五个参数,需要自己修改一下
if (mysql_real_connect(&conn, "localhost", "root", "p@s#0fSPV", "pvault", , NULL, )) {
printf("coneect mysql successful\n");
char insert_query[];
//insert
memset(insert_query, , sizeof(insert_query));
strcat(insert_query, "insert into user(name,password) values('");
strcat(insert_query, per.name);
strcat(insert_query, "','");
strcat(insert_query, per.passwd);
strcat(insert_query, "')");
printf("SQL语句: %s\n", insert_query);
res = mysql_query(&conn, insert_query);
if (!res) {
printf("insert %lu rows\n", (unsigned long)mysql_affected_rows(&conn));
sprintf(msg_stoc.check,"%s","no");
}
else {
printf("insert error\n");
} }
} int main()
{
msgid_ctos = get_ctos_msg();
msgid_stoc = get_stoc_msg(); //int sockfd = socket_rcv();
while()
{
int n;
long type;
//per.type = 1;
msgrcv(msgid_ctos,&per,sizeof(per)-sizeof(type),,);
printf("name:%s,passwd:%s\n", per.name,per.passwd);
sleep();
open_cli();
//Msg m;
msg_stoc.type = ;
sprintf(msg_stoc.check,"%s","no");
printf("check:%s", msg_stoc.check);
msgsnd(msgid_stoc,&msg_stoc,sizeof(msg_stoc)-sizeof(type),);
}
}

  登录也是通过消息队列接收用户名和密码,并从查询出数据,进行对比,是否可以登录,代码如下:

  

#include"my.h"

Msg per;
Msg_stoc msg_stoc; static int msgid_ctos;
static int msgid_stoc; void login(void)
{
MYSQL conn;
int res;
MYSQL_RES * result;
MYSQL_ROW row;
mysql_init(&conn); //第三、四和五个参数,需要自己修改一下
if (mysql_real_connect(&conn, "localhost", "root", "p@s#0fSPV", "pvault", , NULL, )) {
printf("coneect mysql successful\n");
char select_query[] = {};
snprintf(select_query,,"select name,password from user where name='%s'",per.name);
printf("SQL语句: %s\n", select_query);
if (mysql_query(&conn, select_query) != ) {
fprintf(stderr, "查询失败\n");
exit();
}
else {
if ((result = mysql_store_result(&conn)) == NULL) {
fprintf(stderr, "保存结果集失败\n");
exit();
}
else {
while ((row = mysql_fetch_row(result)) != NULL) {
printf("name is %s , ", row[]);
printf("age is %s\n", row[]);
if(( == strcmp(row[],per.name)) && ( == strcmp(row[],per.passwd)))
strcpy(per.info,"login success!");
}
}
} }
} int main()
{
msgid_ctos = get_ctos_msg();
msgid_stoc = get_stoc_msg(); //int sockfd = socket_rcv();
while()
{
int n;
long type;
//per.type = 3;
msgrcv(msgid_ctos,&per,sizeof(per)-sizeof(type),,);
//printf("name:%s,passwd:%s\n", per.name,per.passwd);
sleep();
login();
//Msg m;
msg_stoc.type = ;
sprintf(msg_stoc.check,"%s","yes");
printf("check:%s", per.info);
msgsnd(msgid_stoc,&msg_stoc,sizeof(msg_stoc)-sizeof(type),);
}
}

  

  聊天是通过socket(UDP)实现的。

  总结:通过做这种小项目学到了很多,也发现许多不足,最重要的就是架构能力,之前都是做一小块,没有大局观,虽然项目小,但五张俱全,很锻炼人,继续找项目做!

  

消息队列、socket(UDP)实现简易聊天系统的更多相关文章

  1. 用消息队列和socket实现聊天系统

    前言:最近在学进程间通信,所以做了一个小项目练习一下.主要用消息队列和socket(UDP)实现这个系统,并数据库存储数据,对C语言操作不熟悉的可以参照我的这篇博客:https://www.cnblo ...

  2. 7月目标 socket , 一致性哈希算法 ; mongodb分片; 分布式消息队列; 中间件的使用场景

      分布式的基础:一致性哈希  路由算法的一致性hash http://www.jiacheo.org/blog/174 http://www.tuicool.com/articles/vQVbmai ...

  3. 【windows 操作系统】进程间通信(IPC)简述|无名管道和命名管道 消息队列、信号量、共享存储、Socket、Streams等

    一.进程间通信简述 每个进程各自有不同的用户地址空间,任何一个进程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内核,在内核中开辟一块缓冲区,进程1把数据从用户空间拷到内核缓冲区,进 ...

  4. .NetCore利用BlockingCollection实现简易消息队列

    前言 消息队列现今的应用场景越来越大,常用的有RabbmitMQ和KafKa. 我们用BlockingCollection来实现简单的消息队列. 实现消息队列 用Vs2017创建一个控制台应用程序.创 ...

  5. 【网络编程基础】Linux下进程通信方式(共享内存,管道,消息队列,Socket)

    在网络课程中,有讲到Socket编程,对于tcp讲解的环节,为了加深理解,自己写了Linux下进程Socket通信,在学习的过程中,又接触到了其它的几种方式.记录一下. 管道通信(匿名,有名) 管道通 ...

  6. 8.7 进程间的通讯:管道、消息队列、共享内存、信号量、信号、Socket

    进程间的通讯 进程间为什么需要通讯? 共享数据.数据传输.消息通知.进程控制 进程间的通讯有哪些类型? 首先,联系前面讲过的知识,进程之间的用户地址空间是相互独立的,不能进行互相访问,但是,内核空间却 ...

  7. 持久化消息队列memcacheq的安装配置

    MemcacheQ 是一个基于 MemcacheDB 的消息队列服务器. 一.memcacheq介绍 特性: 1.简单易用 2.处理速度快 3.多条队列 4.并发性能好 5.与memcache的协议兼 ...

  8. 深入浅出 消息队列 ActiveMQ(转)

    一. 概述与介绍 ActiveMQ 是Apache出品,最流行的.功能强大的即时通讯和集成模式的开源服务器.ActiveMQ 是一个完全支持JMS1.1和J2EE 1.4规范的 JMS Provide ...

  9. 007. 服务间通信 RPC & REST over HTTP(s) & 消息队列

    服务间通信 服务间通信的几种方式: RPC.REST over HTTP(s).消息队列.  https://www.jianshu.com/p/2a01d4383d0b RPC https://bl ...

随机推荐

  1. <玩转Django2.0>读书笔记:表单

    1. 表单字段 参考: 官方文档 Django表单字段汇总 2. 表单代码示例(forms.Form) # form.py代码 # 获取数据库数据 choices_list = [(i+1,v['ty ...

  2. Lightscape

    Lightscape是一种先进的光照模拟和可视化设计系统,用于对三维模型进行精确的光照模拟和灵活方便的可视化设计. Lightscape是一个光照渲染软件,它特有的光能传递计算方式和材质属性所产生的独 ...

  3. PBRT笔记(14)——光线传播2:体积渲染

    传输公式 传输方程是控制光线在吸收.发射和散射辐射的介质中的行为的基本方程.它解释了第11章中描述的所有体积散射过程--吸收.发射和内.外散射.并给出了一个描述环境中辐射分布的方程.光传输方程实际上是 ...

  4. 2018-2019-2 网络对抗技术 20162329 Exp1 PC平台逆向破解

    目录 1.实践目标 2.实验内容 2.1 手工修改可执行文件,改变程序执行流程,直接跳转到getShell函数. 2.2 利用foo函数的Bof漏洞,构造一个攻击输入字符串,覆盖返回地址,触发getS ...

  5. BZOJ1431 : MLand

    考虑任意一棵生成树,它的代价是一个一次函数. 因此所有生成树的最小值随着时间变化呈现出的是一个上凸壳. 三分查找最大值即可. 时间复杂度$O(m\log m\log w)$. #include< ...

  6. Hive表生成函数explode讲解

    Hive中的表分析函数接受零个或多个输入,然后产生多列或多行输出. 1.explode函数 explode函数以array类型数据输入,然后对数组中的数据进行迭代,返回多行结果,一行一个数组元素值 A ...

  7. centos查看apache用的是哪个httpd.conf

    httpd -V得到类似如下结果: -D HTTPD_ROOT="/etc/httpd" -D SERVER_CONFIG_FILE="conf/httpd.conf&q ...

  8. vscode1.30.1使用的electron3.0.10中的bug

    PS C:\GitHub\vscode2\vscode> scripts\code.bat[13:07:19] Syncronizing built-in extensions...[13:07 ...

  9. 用Java写hello world

    public class HelloWorld{ public static void main(String[] args){ System.out.println("hello worl ...

  10. Linux shell编程— 命令替换

    有两种方法可以将命令输出赋给变量 反引号字符(`) $()格式 命令替换允许你将shell 命令的输出赋给变量 要么用一对反引号把整个命令行围起来: testing=`data` 要么使用$()格式 ...