UDP -- 简易聊天室

server/2025/1/11 15:48:58/

目录

gitee(内有详细代码)

图解

MessageRoute.hpp

UdpClient.hpp

UdpServer.hpp

Main.hpp

运行结果(本地通信)

如何分开对话显示?


gitee(内有详细代码)

chat_room · zihuixie/Linux_Learning - 码云 - 开源中国icon-default.png?t=O83Ahttps://gitee.com/zihuixie/linux_-learning/tree/master/chat_room

图解

MessageRoute.hpp

#pragma once
#include <iostream>
#include <vector>
#include <string>
#include <vector>
#include <sys/types.h>
#include <sys/socket.h>
#include <functional>
#include "InetAddr.hpp"
#include "ThreadPool.hpp"
#include "LockGuard.hpp"// 通信模块:实现消息的转发
//只要有人发消息,就把该消息转发给所有的在线用户using task_t =std::function<void()>;class MessageRoute
{
private:// 判断地址是否已保存数组中bool IsExists(const InetAddr &addr){for (auto a : _online_user){if (a == addr)return true;}return false;}public:MessageRoute(){pthread_mutex_init(&_mutex,nullptr);}~MessageRoute(){pthread_mutex_destroy(&_mutex);}// 添加用户void Adduser(const InetAddr &addr){LockGuard lock(&_mutex);//保护数组临界资源// 已存在则不添加if (IsExists(addr))return;_online_user.push_back(addr);}// 删除用户void Deluser(const InetAddr &addr){LockGuard lock(&_mutex);//保护数组临界资源// 不存在该用户,不需要删除if (!IsExists(addr))return;// 用迭代器删除for (auto iter = _online_user.begin(); iter != _online_user.end(); iter++){if (*iter == addr){_online_user.erase(iter);break;}}}//转发消息void RouteHelper(int sockfd,std::string message,InetAddr who){LockGuard lock(&_mutex);for(auto user:_online_user){//设置要发送的消息std::string send_message ="\n["+who.Ip()+":"+std::to_string(who.Port())+"]# "+message+"\n";struct sockaddr_in clientaddr=user.Addr();::sendto(sockfd,send_message.c_str(),send_message.size(),0,(struct sockaddr*)&clientaddr,sizeof(clientaddr));}}//通信模块//哪个套接字,发了什么消息,谁发的void Route(int sockfd,std::string message,InetAddr who){Adduser(who);//该用户要退出,删除该用户if(message=="Q" || message=="QUIT")Deluser(who);task_t t=std::bind(&MessageRoute::RouteHelper,this,sockfd,message,who);//让线程池来转发ThreadPool<task_t>::GetInstance()->Enqueue(t);}
private:pthread_mutex_t _mutex;std::vector<InetAddr> _online_user; // 用数组存储在线用户(用地址代表用户)
};

UdpClient.hpp

#include <iostream>
#include <string>
#include <cstring>
#include <cstdlib>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "Thread.hpp"using namespace ThreadModule;void Usage(std::string proc)
{std::cout << "Usage:\n\t" << proc << " serverip serverport\n"<< std::endl;
}int InitClient(std::string &serverip, uint16_t serverport, struct sockaddr_in *server)
{int sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (sockfd < 0){std::cerr << "socket error" << std::endl;return -1;}// 初始化地址memset(server, 0, sizeof(struct sockaddr_in));server->sin_family = AF_INET;server->sin_port = htons(serverport);server->sin_addr.s_addr = inet_addr(serverip.c_str());return sockfd;
}// 接收消息
void recvmessage(int sockfd, std::string name)
{while (true){// 发送方的地址struct sockaddr_in peer;socklen_t len = sizeof(peer);char buffer[1024];// 获取接收方的地址ssize_t n = recvfrom(sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&peer, &len);if (n > 0){buffer[n] = 0;//向标准错误中输出(标准错误也是个文件描述符)fprintf(stderr, "[%s]%s\n", name.c_str(), buffer);}}
}void sendmessage(int sockfd, struct sockaddr_in &server, std::string name)
{std::string message;while (true){printf("%s | Enter# ", name.c_str());fflush(stdout);std::getline(std::cin, message);// 传接收方的地址sendto(sockfd, message.c_str(), sizeof(message), 0, (struct sockaddr *)&server, sizeof(server));}
}int main(int argc, char *argv[])
{if (argc != 3){Usage(argv[0]);exit(1);}//填充地址struct sockaddr_in serveraddr;std::string serverip = argv[1];uint16_t serverport = std::stoi(argv[2]);int sockfd = InitClient(serverip, serverport, &serveraddr);if (sockfd == -1){return 1;}//定义函数并绑定参数func_t r = std::bind(&recvmessage, sockfd, std::placeholders::_1);func_t s = std::bind(&sendmessage, sockfd, serveraddr, std::placeholders::_1);//创建两个线程:发消息 && 收消息//就可以同时实现收消息 && 发消息 -- 全双工//如果收消息 和 发消息 都由一个线程实现 -- 半双工Thread Recver(r, "recver");Thread Sender(s, "sender");//启动线程Recver.Start();Sender.Start();Recver.Join();Sender.Join();return 0;
}

UdpServer.hpp

#pragma once#include <iostream>
#include <string>
#include <cerrno>
#include <cstring>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <functional>
#include "InetAddr.hpp"
#include "Log.hpp"enum
{SOCKET_ERROR = 1, // 套接字创建失败BIND_ERROR,       // 绑定失败USAGE_ERROR       // 用法错误
};const static int defaultfd = -1; // 默认描述符的值为-1// 接收任务
using hander_message_t = std::function<void(int sockfd, const std::string message, const InetAddr who)>;class UdpServer
{
public:// 外界需要传端口号UdpServer(uint16_t port, hander_message_t hander_message): _port(port), _sockfd(defaultfd), _isrunning(false), _hander_message(hander_message){}void InitServer(){// 创建套接字_sockfd = socket(AF_INET, SOCK_DGRAM, 0);if (_sockfd < 0) // 创建失败{LOG(FATAL, "socket error, %s, %d\n", strerror(errno), errno);exit(SOCKET_ERROR);}LOG(INFO, "socket create success, sockfd: %d\n", _sockfd);// 填 sockaddr_instruct sockaddr_in local;     // 在栈上开辟空间bzero(&local, sizeof(local)); // 将内存设置为全0local.sin_family = AF_INET;         // 协议族(用哪个协议)local.sin_port = htons(_port);      // 端口号,转为网络序列(大端)local.sin_addr.s_addr = INADDR_ANY; // IP地址(任意地址)int n = bind(_sockfd, (struct sockaddr *)&local, sizeof(local));// 绑定失败if (n < 0){LOG(FATAL, "bind error,%s,%d\n", strerror(errno), errno);exit(BIND_ERROR);}LOG(INFO, "socket bind success\n");}// 启动服务器,服务器只接收信息和做应答void Start(){_isrunning = true;while (true){char message[1024];struct sockaddr_in peer; // 发送方的地址socklen_t len = sizeof(peer);// 接收数据报,返回值为接收到的字节数ssize_t n = recvfrom(_sockfd, message, sizeof(message) - 1, 0, (struct sockaddr *)&peer, &len);if (n > 0){message[n] = 0;InetAddr addr(peer); // 发送方的IP和端口号LOG(DEBUG, "get message from [%s:%d]:%s\n", addr.Ip().c_str(), addr.Port(), message);//转发消息_hander_message(_sockfd,message,addr);}_isrunning = false;}}~UdpServer(){}private:int _sockfd;uint16_t _port;bool _isrunning;hander_message_t _hander_message;
};

Main.hpp

#include<iostream>
#include<memory>
#include"UdpServer.hpp"
#include"MessageRoute.hpp"void Usage(std::string proc)
{std::cout<<"Usage:\n\t"<<proc<<" local_port\n"<<std::endl;
}int main(int argc,char* argv[])
{if(argc!=2){Usage(argv[0]);exit(USAGE_ERROR);}EnableScreen();MessageRoute route;//创建一个实例uint16_t port=std::stoi(argv[1]);std::unique_ptr<UdpServer> usvr=std::make_unique<UdpServer>(port,std::bind(&MessageRoute::Route,&route,std::placeholders::_1,std::placeholders::_2,std::placeholders::_3));usvr->InitServer();usvr->Start();return 0;}

运行结果(本地通信)

如何分开对话显示?

在启动客户端时,打开两个对话,输入命令 ./udpclient 127.0.0.1 8080 2>/dev/pts/2


http://www.ppmy.cn/server/157508.html

相关文章

谷歌浏览器的开发者文档与资源

谷歌浏览器不仅是全球最流行的web浏览器之一&#xff0c;还提供了功能强大的开发者工具和丰富的开发资源。本文将详细介绍如何利用这些工具和资源来 提升你的网页开发效率。 一、Chrome DevTools简介 Chrome DevTools是一套内置于谷歌浏览器中的开发者工具&#xff0c;允许开发…

国产linux系统(银河麒麟,统信uos)使用 PageOffice 实现后台生成单个PDF文档

PageOffice 国产版 &#xff1a;支持信创系统&#xff0c;支持银河麒麟V10和统信UOS&#xff0c;支持X86&#xff08;intel、兆芯、海光等&#xff09;、ARM&#xff08;飞腾、鲲鹏、麒麟等&#xff09;、龙芯&#xff08;LoogArch&#xff09;芯片架构。 PageOffice 版本&…

计算机网路HTTP、TCP详解

HTTP HTTP基本概念 HTTP&#xff08;超文本传输协议&#xff09;&#xff1a;HTTP是在计算机世界中两点之间传输文字、图片、视频等超文本内容数据的约束与规范。 常见状态码&#xff1a; 2xx&#xff1a;报文被收到&#xff0c;已经在正确处理中。 3xx&#xff1a;重定向…

编写python的几种方式

编写python的三种方式&#xff1a; 1、在IDLE中直接编写 但这个方法是单命令执行&#xff0c;也就是编写一行后按回车键就会运行一下&#xff0c;适用于简单计算&#xff0c;不适用于编写多行代码 2、进入IDLE后新建一个编写页面&#xff0c;这个页面可以编写多行代码并统一执…

Spring MVC详细介绍

1.MVC 设计模式 MVC&#xff08;Model-View-Controller&#xff09;是一种常见的软件设计模式&#xff0c;用于将应用程序的逻辑分离成三个独立的组件&#xff1a; 模型&#xff08;Model&#xff09;&#xff1a;模型是应用程序的数据和业务逻辑的表示。它负责处理数据的读取…

vue实现点击按钮复制文本

在Vue项目中&#xff0c;实现点击按钮复制文本到剪贴板有多种方法。以下是三种常见的实现方案&#xff1a; 使用原生API 可以使用原生的JavaScript API来实现复制功能。以下是一个简单的示例&#xff1a; <template><div><button click"copyText"&…

PyCharm 的安装与使用(Window)

1 PyCharm 简介 PyCharm 是一款由 JetBrains 公司开发的专门用于 Python 语言开发的集成开发环境&#xff08;IDE&#xff09;。以下是其相关介绍&#xff1a; 1.1 特点与功能 智能代码编辑&#xff1a;提供高度智能化的代码编辑器&#xff0c;支持语法高亮、自动补全、代码重…

互联网生活中的隐私保护:用隐私换便利还是花钱护隐私?

隐私与便利&#xff1a;数字世界的选择 在这个数字化的时代&#xff0c;你是否曾经停下来思考过&#xff0c;当你享受快捷便利的网络服务时&#xff0c;自己的隐私究竟付出了什么代价&#xff1f;一个无处不在的事实是&#xff0c;我们的个人信息每天都在被收集、分析&#xf…