当前位置: 首页 > news >正文

9.9网编项目——UDP网络聊天室

服务器端(服务器广播未实现)

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <errno.h>
#include <unistd.h>
#include <25061head.h>
#define SER_IP "192.168.144.128"
#define SER_PORT 8888// 链表节点结构定义
typedef struct Node
{char usrName[30];           // 用户名struct sockaddr_in cin;     // 用户的地址信息struct Node *next;          // 指针域
}*linklist;// 消息结构定义
struct msgTyp
{char type;char usrName[30];char msgText[50];
};int main(int argc, const char *argv[])
{// 前期配置int sfd = socket(AF_INET, SOCK_DGRAM, 0);if(sfd == -1){ERR_MSG("socket error");return -1;}struct sockaddr_in sin;sin.sin_family = AF_INET;sin.sin_port = htons(SER_PORT);sin.sin_addr.s_addr = inet_addr(SER_IP);if(bind(sfd, (struct sockaddr*)&sin, sizeof(sin)) == -1){ERR_MSG("bind error");close(sfd);return -1;}printf("绑定成功\n");struct sockaddr_in cin;socklen_t addrlen = sizeof(cin);struct msgTyp recv_msg;  // 接收的信息// 创建头节点并初始化linklist head = (linklist)malloc(sizeof(struct Node));if(head == NULL){ERR_MSG("malloc error");close(sfd);return -1;}head->next = NULL;  // 初始化头节点// 核心操作while(1){// 清空接收缓冲区memset(&recv_msg, 0, sizeof(recv_msg));// 接收消息ssize_t recv_len = recvfrom(sfd, &recv_msg, sizeof(recv_msg), 0, (struct sockaddr*)&cin, &addrlen);if(recv_len == -1){ERR_MSG("recvfrom error");continue;}switch(recv_msg.type){case 'L':  // login登录printf("%s登录成功\n", recv_msg.usrName);// 创建新用户的节点并初始化linklist temp = (linklist)malloc(sizeof(struct Node));if(temp == NULL){ERR_MSG("malloc error");break;} //存储新用户结点的相关信息strcpy(temp->usrName, recv_msg.usrName);temp->cin = cin;//头插temp->next = head->next;head->next = temp;// 广播登录消息linklist s = head->next;char sbuf[128] = "";snprintf(sbuf, sizeof(sbuf)-1, "%s已上号!", recv_msg.usrName);strcpy(recv_msg.msgText,sbuf);//放入recv.msg中,发给其他人//recv_msg.msgText[sizeof(recv_msg.msgText)-1] = '\0';while(s != NULL){// 跳过发送者自己(用IP和端口区分,后面的广播都是这种方法排除自己)if (s->cin.sin_port == cin.sin_port && s->cin.sin_addr.s_addr == cin.sin_addr.s_addr){s = s->next;continue;}sendto(sfd, &recv_msg, sizeof(recv_msg), 0, (struct sockaddr*)&(s->cin), sizeof(s->cin));s = s->next;}break;case 'C':  // chat聊天printf("%s:chat成功\n", recv_msg.usrName);char cbuf[128] = "";snprintf(cbuf, sizeof(cbuf)-1, "%s说:%s", recv_msg.usrName, recv_msg.msgText);strcpy(recv_msg.msgText, cbuf);//recv_msg.msgText[sizeof(recv_msg.msgText)-1] = '\0';s = head->next;//从第一个用户开始while(s != NULL){// 跳过发送者自己if (s->cin.sin_port == cin.sin_port && s->cin.sin_addr.s_addr == cin.sin_addr.s_addr){s = s->next;continue;}sendto(sfd, &recv_msg, sizeof(recv_msg), 0, (struct sockaddr*)&(s->cin), sizeof(s->cin));s = s->next;}break;case 'Q':  // quit退出printf("%s退出聊天室\n", recv_msg.usrName);char qbuf[128] = "";snprintf(qbuf, sizeof(qbuf)-1, "%s退出聊天室", recv_msg.usrName);strcpy(recv_msg.msgText, qbuf);//recv_msg.msgText[sizeof(recv_msg.msgText)-1] = '\0';// 先广播退出消息s = head->next;while(s != NULL){// 跳过发送者自己if (s->cin.sin_port == cin.sin_port && s->cin.sin_addr.s_addr == cin.sin_addr.s_addr){s = s->next;continue;}sendto(sfd, &recv_msg, sizeof(recv_msg), 0, (struct sockaddr*)&(s->cin), sizeof(s->cin));s = s->next;}           // 然后从链表中删除该用户节点linklist prev = head;linklist curr = head->next;int found = 0;//判断找没找到结点while(curr != NULL){if(strcmp(curr->usrName, recv_msg.usrName) == 0){//头删prev->next = curr->next;free(curr);curr=NULL;found = 1;printf("已从链表中移除用户: %s\n", recv_msg.usrName);break;}prev = curr;curr = curr->next;}if(found==0){printf("警告: 未在链表中找到用户 %s\n", recv_msg.usrName);}break;default:printf("发送格式错误\n");}}close(sfd);return 0;
}

客户端

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <errno.h>
#include <unistd.h>
#include <pthread.h>
#include <25061head.h>
#define SER_IP "192.168.144.128"
#define SER_PORT 8888struct msgTyp
{char type;char usrName[30];char msgText[50];
};int cfd;                  // 客户端socket
struct sockaddr_in sin;   // 服务器的相关配置
char usrName[30] = "";   // 用户名
//虽然是线程,但有多个阻塞函数,所以用多线程
//该线程主要功能是接收消息
void *recv_msg_thread(void *arg)
{struct msgTyp recv_msg;socklen_t addrlen = sizeof(sin);while(1){// 清空接收缓冲区memset(&recv_msg, 0, sizeof(recv_msg));// 接收服务器发送的消息ssize_t recv_len = recvfrom(cfd, &recv_msg,sizeof(recv_msg),0,(struct sockaddr*)&sin, &addrlen);// 打印接收的消息printf("%s\n", recv_msg.msgText);fflush(stdout);  // 刷新输出缓冲区}// 接收线程退出时关闭socketclose(cfd);pthread_exit(0);return NULL;
}int main(int argc, const char *argv[])
{// 获取用户名printf("请输入你的用户名: ");fgets(usrName, sizeof(usrName)-1, stdin);// 去除换行符usrName[strcspn(usrName, "\n")] = '\0';// 创建socketcfd = socket(AF_INET, SOCK_DGRAM, 0);if(cfd == -1){ERR_MSG("socket error");return -1;}    // 初始化服务器相关配置sin.sin_family = AF_INET;sin.sin_port = htons(SER_PORT);sin.sin_addr.s_addr = inet_addr(SER_IP);// 配置登录消息struct msgTyp send_msg;send_msg.type = 'L';strcpy(send_msg.usrName,usrName);//strcpy(send_msg.msgText, "");if(sendto(cfd, &send_msg, sizeof(send_msg), 0,(struct sockaddr*)&sin, sizeof(sin)) == -1){ERR_MSG("sendto error");close(cfd);return -1;}printf("登录成功!输入'quit'退出聊天室...\n");// 创建接收消息的线程pthread_t recv_tid;if(pthread_create(&recv_tid, NULL, recv_msg_thread, NULL) != 0){ERR_MSG("pthread_create error");close(cfd);return -1;}// 分离线程,系统自动回收pthread_detach(recv_tid);// 主线程用于发送消息while(1){char input[50] = "";fgets(input, sizeof(input)-1, stdin);// 去除换行符input[strcspn(input, "\n")] = '\0';//输入quit,消息类型则设置从Q类型if(strcmp(input, "quit") == 0){send_msg.type = 'Q';strcpy(send_msg.usrName, usrName); sendto(cfd, &send_msg, sizeof(send_msg),0,(struct sockaddr*)&sin, sizeof(sin));printf("已退出聊天室\n");close(cfd);return 0;}// 发送聊天消息send_msg.type = 'C';strcpy(send_msg.usrName, usrName);strcpy(send_msg.msgText, input);if(sendto(cfd, &send_msg, sizeof(send_msg),0,(struct sockaddr*)&sin, sizeof(sin)) == -1){ERR_MSG("sendto error");close(cfd);return -1;}}//关闭socketclose(cfd);return 0;
}


文章转载自:

http://035kCdXD.ndngj.cn
http://mXPfr1bY.ndngj.cn
http://3iZjvia9.ndngj.cn
http://grNi1bp9.ndngj.cn
http://qnMFgcFD.ndngj.cn
http://Fp2eChlc.ndngj.cn
http://VR9nsj2E.ndngj.cn
http://Z5MtGXRJ.ndngj.cn
http://pFI7weog.ndngj.cn
http://qayU51l8.ndngj.cn
http://Xx6hBv45.ndngj.cn
http://4dY4Cb2x.ndngj.cn
http://FGeHDRq4.ndngj.cn
http://liD2Yxt5.ndngj.cn
http://h7NZ4G9n.ndngj.cn
http://rVVVnQlR.ndngj.cn
http://1er7GDqe.ndngj.cn
http://P3Kcmq6Q.ndngj.cn
http://XzQ255En.ndngj.cn
http://A5pTon6K.ndngj.cn
http://AHgjUpVU.ndngj.cn
http://DRV9XGTt.ndngj.cn
http://RQNDKSbU.ndngj.cn
http://W6Hu9kH3.ndngj.cn
http://5ddngBfv.ndngj.cn
http://zeHXw1Uq.ndngj.cn
http://UhLWrmTV.ndngj.cn
http://37Y0tnJD.ndngj.cn
http://CrrEAgBN.ndngj.cn
http://CpdpyH8q.ndngj.cn
http://www.dtcms.com/a/378165.html

相关文章:

  • 单表查询-having和where区别
  • LVGL:基础对象
  • 【LeetCode - 每日1题】将字符串中的元音字母排序
  • 签名、杂凑、MAC、HMAC
  • C++与QT高频面试问题(不定时更新)
  • 数据结构之跳表
  • 记录豆包的系统提示词
  • Docker 从入门到实践:容器化技术核心指南
  • 【Python-Day 43】告别依赖混乱:Python虚拟环境venv入门与实战
  • CF702E Analysis of Pathes in Functional Graph 题解
  • 元宇宙与智慧城市:数字孪生赋能的城市治理新范式
  • es通过分片迁移迁移解决磁盘不均匀问题
  • 深入浅出CRC校验:从数学原理到单周期硬件实现 (2)CRC数学多项式基础
  • 无人设备遥控器之控制指令发送技术篇
  • LinuxC++项目开发日志——高并发内存池(4-central cache框架开发)
  • 解决蓝牙耳机连win11电脑画质依托答辩问题
  • 农业养殖为何离不开温湿度传感器?
  • Android开发 AlarmManager set() 方法与WiFi忘记连接问题分析
  • CKA02-Ingress
  • JavaEE 初阶第二十一期:网络原理,底层框架的“通关密码”(一)
  • TOL-API 基于Token验证文件传输API安全工具
  • 构建一个优雅的待办事项应用:现代JavaScript实践
  • 计算机视觉进阶教学之图像投影(透视)变换
  • 计算机视觉与深度学习 | 基于MATLAB的AI图片识别系统研究
  • 计算机视觉----图像投影(透视)变换(小案例)
  • Docker 学习笔记(七):Docker Swarm 服务管理与 Containerd 实践
  • 3-10〔OSCP ◈ 研记〕❘ WEB应用攻击▸XSS攻击理论基础
  • 微信小程序开发笔记(01_小程序基础与配置文件)
  • ArcGIS JSAPI 高级教程 - ArcGIS Maps SDK for JavaScript - 自定义(GLSL)修改高亮图层样式
  • idea npm install 很慢(nodejs)