鱼C论坛

 找回密码
 立即注册
查看: 2182|回复: 7

[技术交流] C/C++ 网络聊天室 客户端(qt,windows) 服务器(Linux)

[复制链接]
发表于 2021-3-5 14:31:17 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
Qt的学习成果,抛砖引玉,博君一笑

  1. //I/O复用  epoll模型
  2. //服务器代码
  3. #include <stdio.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6. #include <errno.h>
  7. #include <unistd.h>
  8. #include <fcntl.h>
  9. #include <arpa/inet.h>
  10. #include <netinet/in.h>
  11. #include <sys/epoll.h>
  12. #include <sys/socket.h>
  13. #include <sys/types.h>

  14. #define  MAXEVENTS 128

  15. //变量声明;
  16. int userSock[1024];                //套接字数组;
  17. char userId[1024][248]; //id数组;
  18. int maxfd,listenfd;                //最大套接字,监听套接字;

  19. //函数声明;
  20. //套接字初始化;
  21. int IniServer(int port);

  22. //获取xml字符串;
  23. bool GetXmlBuffer(const char *xml,const char *label,char *val);
  24. bool GetXmlBuffer(const char *xml,const char *label,int *val);
  25. bool GetXmlBuffer(const char *xml,const char *label,double *val);

  26. //消息处理;
  27. void dealLogin(int eventfd,char*buffer);
  28. void dealOut(int eventfd,char *buffer);
  29. void dealMail(int evetfd,char *buffer);

  30. //主函数;
  31. int main(int argc,char* argv[])
  32. {
  33.         if(argc !=2 )
  34.         {
  35.                 printf("请指定端口号,例如:./xxx 5081");
  36.                 return -1;
  37.         }

  38.         listenfd = IniServer(atoi(argv[1]));          
  39.         if(listenfd <= 0) {printf("初始化失败\n");return -1;}
  40.        
  41.         memset(userSock,0,sizeof(userSock));
  42.         for(int i = 0;i < 1024;i++) memset(userId[i],0,sizeof(userId[i]));
  43.         userSock[listenfd] = listenfd;
  44.         maxfd = listenfd;

  45.         int epollfd = epoll_create(1);
  46.         struct epoll_event add;
  47.         add.data.fd = listenfd;
  48.         add.events = EPOLLIN;
  49.         epoll_ctl(epollfd,EPOLL_CTL_ADD,listenfd,&add);

  50.         while(1)
  51.         {
  52.                 struct epoll_event events[MAXEVENTS];
  53.                 int infds = epoll_wait(epollfd,events,MAXEVENTS,-1);
  54.                 if(infds < 0) {printf("epoll_wait failed\n");return -1;}
  55.                 if(infds == 0) {printf("超时\n");continue;}

  56.                 for(int i = 0;i < infds;i++)
  57.                 {
  58.                         if(events[i].data.fd == listenfd && (events[i].events&EPOLLIN)) //监听事件;
  59.                         {
  60.                                 struct sockaddr_in clientaddr;
  61.                                 socklen_t len = sizeof(clientaddr);
  62.                                 int clientfd = accept(listenfd,(struct sockaddr*)&clientaddr,&len);
  63.                                 if(clientfd <= 0) {printf("与客户端连接失败\n");continue;}

  64.                                 memset(&add,0,sizeof(add));
  65.                                 add.data.fd = clientfd;
  66.                                 add.events = EPOLLIN;
  67.                                 epoll_ctl(epollfd,EPOLL_CTL_ADD,clientfd,&add);

  68.                                 ntohl(clientaddr.sin_addr.s_addr);
  69.                                 printf("客户端%s连接到服务器\n",inet_ntoa(clientaddr.sin_addr));
  70.                                 userSock[clientfd] = clientfd;

  71.                                 maxfd = clientfd>maxfd?clientfd:maxfd;
  72.                         }

  73.                         else if(events[i].events&EPOLLIN)        //读事件;
  74.                         {
  75.                                 char buffer[1024];
  76.                                 memset(buffer,0,sizeof(buffer));

  77.                                 int iret = recv(events[i].data.fd,buffer,sizeof(buffer),0);
  78.                                 if(iret <= 0)
  79.                                 {
  80.                                         add.data.fd = events[i].data.fd;
  81.                                         add.events = events[i].events;
  82.                                         epoll_ctl(epollfd,EPOLL_CTL_DEL,events[i].data.fd,&add);
  83.                                         close(events[i].data.fd);
  84.                                         printf("socket(%d)已断开连接\n",events[i].data.fd);
  85.                                 }
  86.                                
  87.                                 else
  88.                                 {
  89.                                         printf("接收:%s\n",buffer);
  90.                                         char val[1024] = {0};

  91.                                         if(GetXmlBuffer(buffer,"login",val))
  92.                                         {
  93.                                                 dealLogin(events[i].data.fd,buffer);
  94.                                                 continue;
  95.                                         }
  96.                                         else if(GetXmlBuffer(buffer,"out",val))
  97.                                         {
  98.                                                 dealOut(events[i].data.fd,buffer);
  99.                                                 continue;
  100.                                         }
  101.                                         else if(GetXmlBuffer(buffer,"mail",val))
  102.                                         {
  103.                                                 dealMail(events[i].data.fd,buffer);       
  104.                                                 continue;
  105.                                         }
  106.                                         //printf("接收:%s\n",buffer);
  107.                                         //send(events[i].data.fd,buffer,sizeof(buffer),0);
  108.                                 }
  109.                         }
  110.                 }
  111.         }
  112. }

  113. //函数定义;
  114. int IniServer(int port)
  115. {
  116.         int sockfd = socket(AF_INET,SOCK_STREAM,0);
  117.         if(sockfd <= 0) return -1;
  118.         struct sockaddr_in servaddr;
  119.         servaddr.sin_family = AF_INET;
  120.         servaddr.sin_port = htons(port);
  121.         servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  122.         if(bind(sockfd,(struct sockaddr*)&servaddr,sizeof(servaddr))!=0) return -1;
  123.         if(listen(sockfd,5)!=0) return -1;
  124.         return sockfd;
  125. }

  126. bool GetXmlBuffer(const char *xml,const char *label,char *val)
  127. {
  128.         char *start,*end;
  129.         char label_start[51],label_end[51];
  130.         memset(label_start,0,sizeof(label_start));
  131.         memset(label_start,0,sizeof(label_end));
  132.         int label_len = strlen(label);
  133.         sprintf(label_start,"<%s>",label);
  134.         sprintf(label_end,"</%s>",label);
  135.         start = (char*)strstr(xml,label_start),end = (char*)strstr(xml,label_end);
  136.         if(start == NULL || end == NULL) return false;
  137.         strncpy(val,start+label_len+2,end-start-label_len-2);
  138.         return true;
  139. }

  140. bool GetXmlBuffer(const char *xml,const char *label,int *val)
  141. {
  142.         char strval[51];
  143.         if(GetXmlBuffer(xml,label,strval) == false) return false;
  144.         *val = atoi(strval);
  145.         return true;
  146. }

  147. bool GetXmlBuffer(const char *xml,const char *label,double *val)
  148. {
  149.         char strval[51];
  150.         if(GetXmlBuffer(xml,label,strval) == false) return false;
  151.         *val = atof(strval);
  152.         return true;
  153. }

  154. void dealLogin(int eventfd,char *buffer)
  155. {
  156.         char tmp[1024] = {0};
  157.         for(int i = 0;i <= maxfd;i++)
  158.         {
  159.                 if(userSock[i] == 0) continue;
  160.                 if(userSock[i] == listenfd) continue;

  161.                 send(userSock[i],buffer,strlen(buffer),0);
  162.         }
  163.         for(int i = 0;i <= maxfd;i++)
  164.         {
  165.                 if(userSock[i] == 0) continue;
  166.                 if(userSock[i] == listenfd) continue;
  167.                 if(userSock[i] == eventfd) continue;

  168.                 memset(tmp,0,sizeof(tmp));
  169.                 sprintf(tmp,"<login>%s</login>",userId[userSock[i]]);
  170.                 send(eventfd,tmp,strlen(tmp),0);
  171.                 usleep(90000);
  172.         }

  173.         memset(tmp,0,sizeof(tmp));
  174.         GetXmlBuffer(buffer,"login",userId[eventfd]);
  175. }

  176. void dealOut(int eventfd,char *buffer)
  177. {
  178.         for(int i = 0;i <= maxfd;i++)
  179.         {
  180.                 if(userSock[i] == 0) continue;
  181.                 if(userSock[i] == listenfd) continue;
  182.                 if(userSock[i] == eventfd) continue;
  183.                
  184.                 send(userSock[i],buffer,strlen(buffer),0);
  185.         }

  186.         for(int i = maxfd;i >= 0;i--)
  187.         {
  188.                 if(userSock[i] == 0) continue;
  189.                 maxfd = userSock[i];
  190.                 break;
  191.         }

  192.         userSock[eventfd] = 0;
  193.         memset(userId[eventfd],0,sizeof(userId[eventfd]));
  194. }

  195. void dealMail(int eventfd,char *buffer)
  196. {
  197.         char val[1024] = {0};
  198.         GetXmlBuffer(buffer,"mail",val);
  199.         char tmp[2048] = {0};
  200.         sprintf(tmp,"<mail>---------%s---------\n%s\n----------------------</mail>",\
  201.                                                                                                                                 userId[eventfd],val);
  202.         for(int i = 0;i <= maxfd;i++)
  203.         {
  204.                 if(userSock[i] == 0) continue;
  205.                 if(userSock[i] == listenfd) continue;
  206.                
  207.                 send(userSock[i],tmp,strlen(tmp),0);
  208.         }
  209. }
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2021-3-5 14:33:25 | 显示全部楼层
本帖最后由 弦歌雅意 于 2021-3-5 18:02 编辑

客户端代码(启动界面)


****************************************
头文件

  1. #ifndef MAINWINDOW_H
  2. #define MAINWINDOW_H

  3. #include <QMainWindow>
  4. #include <QTcpSocket>
  5. #include "talk.h"

  6. QT_BEGIN_NAMESPACE
  7. namespace Ui { class MainWindow; }
  8. QT_END_NAMESPACE

  9. class MainWindow : public QMainWindow
  10. {
  11.     Q_OBJECT

  12. public:
  13.     MainWindow(QWidget *parent = nullptr);
  14.     ~MainWindow();
  15.     talk t;

  16. private slots:
  17.     void on_startbtn_clicked();

  18. private:
  19.     Ui::MainWindow *ui;
  20. };
  21. #endif // MAINWINDOW_H
复制代码


******************************************
源文件

  1. #include "mainwindow.h"
  2. #include "ui_mainwindow.h"
  3. #include <QString>
  4. #include <QHostAddress>
  5. #include <QTextEdit>
  6. #include <QKeyEvent>

  7. MainWindow::MainWindow(QWidget *parent)
  8.     : QMainWindow(parent)
  9.     , ui(new Ui::MainWindow)
  10. {
  11.     ui->setupUi(this);

  12.     this->setWindowTitle("启动程序");

  13.     connect(&t,&talk::Mysignal,
  14.             [=]()
  15.             {
  16.                 this->show();
  17.             });
  18. }

  19. MainWindow::~MainWindow()
  20. {
  21.     delete ui;
  22. }


  23. void MainWindow::on_startbtn_clicked()
  24. {
  25.     QString ipstr;
  26.     qint16 port;

  27.     t.id = ui->idline->text();
  28.     ipstr = ui->ipline->text();
  29.     port = ui->portline->text().toInt();

  30.     t.sockfd->connectToHost(QHostAddress(ipstr),port);
  31.     this->hide();
  32.     t.show();
  33. }
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-3-5 14:33:57 | 显示全部楼层
本帖最后由 弦歌雅意 于 2021-3-5 18:03 编辑

客户端聊天界面

*************************************
头文件

  1. #ifndef TALK_H
  2. #define TALK_H

  3. #include <QWidget>
  4. #include <QTcpSocket>
  5. #include <QString>

  6. namespace Ui {
  7. class talk;
  8. }

  9. class talk : public QWidget
  10. {
  11.     Q_OBJECT

  12. public:
  13.     explicit talk(QWidget *parent = nullptr);
  14.     ~talk();
  15.     QTcpSocket *sockfd;
  16.     QString id;

  17. signals:
  18.     void Mysignal();

  19. private slots:
  20.     void on_pushButton_clicked();

  21.     void on_pushButton_2_clicked();


  22. private:
  23.     Ui::talk *ui;
  24. };

  25. #endif // TALK_H
复制代码


************************************
源文件

  1. #include "talk.h"
  2. #include "ui_talk.h"
  3. #include <QTcpSocket>
  4. #include <QString>
  5. #include <QDebug>
  6. #include <QKeyEvent>
  7. #include <QTextEdit>

  8. //函数声明;
  9. bool GetXmlBuffer(QString &recvStr,QString label);

  10. talk::talk(QWidget *parent) :
  11.     QWidget(parent),
  12.     ui(new Ui::talk)
  13. {
  14.     ui->setupUi(this);

  15.     this->setWindowTitle("聊天板块");

  16.     sockfd = NULL;
  17.     sockfd = new QTcpSocket(this);

  18.     connect(sockfd,&QTcpSocket::connected,
  19.             [=]()
  20.             {
  21.                 ui->readtext->append("和服务器建立连接");
  22.                 QString str = "<login>" + id + "</login>";
  23.                 sockfd->write(str.toUtf8().data());
  24.             });
  25.     connect(sockfd,&QTcpSocket::readyRead,
  26.             [=]()
  27.             {
  28.                 QByteArray array = sockfd->readAll();
  29.                 QString str = QString(array);

  30.                 if(GetXmlBuffer(str,"login"))
  31.                 {

  32.                     ui->idlist->addItem(str);
  33.                 }
  34.                 else if(GetXmlBuffer(str,"out"))
  35.                 {
  36.                     for(int i = 0;i < ui->idlist->count();i++)
  37.                     {
  38.                         if(str == ui->idlist->item(i)->text())
  39.                         {
  40.                             ui->idlist->takeItem(i);
  41.                         }
  42.                     }

  43.                 }
  44.                 else if(GetXmlBuffer(str,"mail"))
  45.                 {
  46.                     ui->readtext->append(str);
  47.                 }
  48.             });
  49. }

  50. talk::~talk()
  51. {
  52.     delete ui;
  53. }

  54. void talk::on_pushButton_clicked()
  55. {
  56.     QString str = ui->writetext->toPlainText();
  57.     str = "<mail>" + str + "</mail>";
  58.     sockfd->write(str.toUtf8().data());
  59.     //ui->readtext->append(str);
  60.     ui->writetext->setText("");
  61. }

  62. void talk::on_pushButton_2_clicked()
  63. {
  64.     QString str = "<out>" + id + "</out>";
  65.     sockfd->write(str.toUtf8().data());
  66.     sockfd->disconnectFromHost();
  67.     ui->readtext->append("已与服务器断开连接");
  68.     this->ui->readtext->setText("");
  69.     this->ui->writetext->setText("");
  70.     this->ui->idlist->clear();
  71.     this->hide();
  72.     emit Mysignal();
  73. }

  74. bool GetXmlBuffer(QString &recvStr,QString label)
  75. {
  76.     QString start = "<" + label + ">";
  77.     QString end = "</" + label + ">";
  78.     int startPos = recvStr.indexOf(start);
  79.     if(startPos == -1) return false;
  80.     recvStr = recvStr.mid(start.length(),recvStr.length()-start.length()-end.length());
  81.     return true;
  82. }



复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-3-5 14:39:48 | 显示全部楼层
本帖最后由 弦歌雅意 于 2021-3-5 14:59 编辑

运行效果如下
1.png
2.png
3.png
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-3-5 14:45:26 | 显示全部楼层
本帖最后由 弦歌雅意 于 2021-3-5 15:13 编辑

我是采用了 epoll模型 来写服务器,并没有使用 多线程/多进程 的方式,所以服务器虽然是在linux运行的,但是和Windows差别不大,改了头文件,再稍微修改一下代码就能在Windows上运行
欢迎各位鱼c的大神提出建议,也欢迎初学的萌新找我交流
以后也许会做一篇详细的教程,包括 基础的socket套接字,以及io复用三种模型(select,poll,epoll)
另外,由于聊天室功能是一点一点完善的,走一步看一步,所以会有一些莫名明其妙的地方,因为修改的不够干净,请各位体谅。

小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-3-5 18:06:11 | 显示全部楼层
最后贴一下ui设计


启动界面

启动界面

聊天界面

聊天界面
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2021-3-5 18:08:38 | 显示全部楼层
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2021-3-5 23:38:55 From FishC Mobile | 显示全部楼层
强啊
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-5-1 05:55

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表