通过网络编程实现一个C/S模式的员工管理系统,服务器端用多进程或多线程模型支持多个客户端同时连接、操作。具体要求如下: (1)员工信息包括三个字段:姓名,年龄,手机号,用结构体表示; (2)服务器功能: 添加新员工——接受客户端发送来的新员工结构体数据,追加到数据文件emp.db中; 获取员工列表——把emp.db中的每个员工结构体数据读出发回客户端; (3)客户端功能: 添加新员工——从终端读取新员工信息并构造为结构体变量,发到服务器请求添加新员工; 显示员工列表——向服务器请求员工列表,接收后显示到终端。 (4)客户端和服务器之间的协议应事先设计,如服务器返回给客户端的各类报告信息、错误信息都应事先统一设计好编号

时间: 2023-05-25 10:05:41 浏览: 71
和格式。 协议设计: 客户端发送数据格式: 添加新员工请求: 编号:0x01 数据:员工结构体 获取员工列表请求: 编号:0x02 数据:无 服务器返回数据格式: 添加新员工成功: 编号:0x11 数据:添加成功 添加新员工失败: 编号:0x12 数据:添加失败原因 获取员工列表成功: 编号:0x21 数据:员工结构体数组 获取员工列表失败: 编号:0x22 数据:获取失败原因 服务器端流程: 1.建立服务器端套接字 2.绑定套接字,并监听客户端连接 3.当有客户端连接时,创建新的子进程处理连接请求 4.子进程接受客户端请求 5.根据客户端请求编号执行相应操作 6.服务器返回数据给客户端 客户端流程: 1.建立客户端套接字 2.连接服务器端 3.从终端读取用户输入,构造请求数据 4.发送请求给服务器端 5.等待服务器返回数据 6.根据服务器返回数据编号执行相应操作 代码实现: 服务器端 ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <errno.h> #include <netinet/in.h> #include <sys/socket.h> #include <sys/types.h> #include <arpa/inet.h> #define PORT 9000 #define MAX_BUFFER 1024 typedef struct _Employee { char name[32]; int age; char phone[16]; } Employee; int addEmployee(Employee employee); Employee* getEmployees(int* count); int main(int argc, char const *argv[]) { int server_fd, client_fd; pid_t child_pid; struct sockaddr_in server_addr, client_addr; socklen_t sin_len = sizeof(client_addr); char buffer[MAX_BUFFER]; int count = 0; server_fd = socket(AF_INET, SOCK_STREAM, 0); if (server_fd < 0) { perror("socket error"); exit(1); } server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = INADDR_ANY; server_addr.sin_port = htons(PORT); if (bind(server_fd, (struct sockaddr*)&server_addr, sizeof(server_addr)) == -1) { perror("bind error"); close(server_fd); exit(1); } if (listen(server_fd, 10) == -1) { perror("listen error"); close(server_fd); exit(1); } printf("Server started.\n"); while (1) { client_fd = accept(server_fd, (struct sockaddr*)&client_addr, &sin_len); if (client_fd == -1) { perror("accept error"); continue; } inet_ntop(AF_INET, &client_addr.sin_addr, buffer, MAX_BUFFER); printf("Client connected: %s\n", buffer); if ((child_pid = fork()) == -1) { perror("fork error"); close(client_fd); continue; } else if (child_pid == 0) { close(server_fd); while (1) { memset(buffer, 0, MAX_BUFFER); int bytes_read = read(client_fd, buffer, MAX_BUFFER); if (bytes_read < 0) { perror("read error"); close(client_fd); exit(1); } else if (bytes_read == 0) { printf("Connection closed.\n"); close(client_fd); exit(0); } int request_code = buffer[0]; if (request_code == 0x01) { Employee employee; memcpy(&employee, buffer + 1, sizeof(Employee)); if (addEmployee(employee) < 0) { strcpy(buffer, "Add employee failed."); write(client_fd, buffer, strlen(buffer)); } else { strcpy(buffer, "Add employee success."); write(client_fd, buffer, strlen(buffer)); } } else if (request_code == 0x02) { Employee* employees = getEmployees(&count); if (employees == NULL || count <= 0) { strcpy(buffer, "Get employees failed."); write(client_fd, buffer, strlen(buffer)); } else { memcpy(buffer, &count, sizeof(count)); memcpy(buffer + sizeof(count), employees, sizeof(Employee) * count); write(client_fd, buffer, sizeof(count) + sizeof(Employee) * count); free(employees); } } } close(client_fd); exit(0); } close(client_fd); } return 0; } int addEmployee(Employee employee) { FILE* fp; fp = fopen("emp.db", "ab"); if (fp == NULL) { perror("open file error"); return -1; } fwrite(&employee, sizeof(Employee), 1, fp); fclose(fp); return 0; } Employee* getEmployees(int* count) { FILE* fp; Employee* employees = NULL; int size = 0; fp = fopen("emp.db", "rb"); if (fp == NULL) { perror("open file error"); return NULL; } fseek(fp, 0, SEEK_END); size = ftell(fp); rewind(fp); if (size % sizeof(Employee) != 0) { perror("file error"); fclose(fp); return NULL; } *count = size / sizeof(Employee); employees = (Employee*)malloc(sizeof(Employee) * (*count)); fread(employees, sizeof(Employee), (*count), fp); fclose(fp); return employees; } ``` 客户端 ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <errno.h> #include <netinet/in.h> #include <sys/socket.h> #include <arpa/inet.h> #define SERVER_IP "localhost" #define PORT 9000 #define MAX_BUFFER 1024 typedef struct _Employee { char name[32]; int age; char phone[16]; } Employee; void addEmployee(int sockfd); void getEmployees(int sockfd); int main(int argc, char const *argv[]) { int sockfd; struct sockaddr_in server_addr; if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) { perror("socket error"); exit(1); } server_addr.sin_family = AF_INET; server_addr.sin_port = htons(PORT); inet_pton(AF_INET, SERVER_IP, &server_addr.sin_addr); if (connect(sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0) { perror("connect error"); exit(1); } char option[16]; while (1) { printf("Enter option: "); scanf("%s", option); if (strcasecmp(option, "add") == 0) { addEmployee(sockfd); } else if (strcasecmp(option, "get") == 0) { getEmployees(sockfd); } else if (strcasecmp(option, "exit") == 0) { break; } else { printf("Invalid option.\n"); } } close(sockfd); return 0; } void addEmployee(int sockfd) { Employee employee; printf("Enter name: "); scanf("%s", employee.name); printf("Enter age: "); scanf("%d", &employee.age); printf("Enter phone: "); scanf("%s", employee.phone); char buffer[MAX_BUFFER]; memset(buffer, 0, MAX_BUFFER); memcpy(buffer, &employee, sizeof(Employee)); buffer[0] = 0x01; write(sockfd, buffer, sizeof(Employee) + 1); memset(buffer, 0, MAX_BUFFER); int bytes_read = read(sockfd, buffer, MAX_BUFFER); if (bytes_read < 0) { perror("read error"); } else { printf("%s\n", buffer); } } void getEmployees(int sockfd) { char buffer[MAX_BUFFER]; memset(buffer, 0, MAX_BUFFER); buffer[0] = 0x02; write(sockfd, buffer, 1); memset(buffer, 0, MAX_BUFFER); int bytes_read = read(sockfd, buffer, MAX_BUFFER); if (bytes_read < 0) { perror("read error"); } else { int count = 0; memcpy(&count, buffer, sizeof(count)); if (count <= 0) { printf("No employee.\n"); } else { Employee* employees = (Employee*)(buffer + sizeof(count)); for (int i = 0; i < count; i++) { printf("name: %s, age: %d, phone: %s\n", employees[i].name, employees[i].age, employees[i].phone); } } } } ```

相关推荐

最新推荐

recommend-type

华为OD机试D卷 - 用连续自然数之和来表达整数 - 免费看解析和代码.html

私信博主免费获取真题解析以及代码
recommend-type

Screenshot_2024-05-10-20-21-01-857_com.chaoxing.mobile.jpg

Screenshot_2024-05-10-20-21-01-857_com.chaoxing.mobile.jpg
recommend-type

数字图像处理|Matlab-频域增强实验-彩色图像的频域滤波.zip

数字图像处理|Matlab-频域增强实验-彩色图像的频域滤波.zip
recommend-type

2024-2030中国定向转向膜市场现状研究分析与发展前景预测报告.docx

2024-2030中国定向转向膜市场现状研究分析与发展前景预测报告
recommend-type

开源工时填报管理系统安装包

开源工时填报管理系统安装包
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

实现实时数据湖架构:Kafka与Hive集成

![实现实时数据湖架构:Kafka与Hive集成](https://img-blog.csdnimg.cn/img_convert/10eb2e6972b3b6086286fc64c0b3ee41.jpeg) # 1. 实时数据湖架构概述** 实时数据湖是一种现代数据管理架构,它允许企业以低延迟的方式收集、存储和处理大量数据。与传统数据仓库不同,实时数据湖不依赖于预先定义的模式,而是采用灵活的架构,可以处理各种数据类型和格式。这种架构为企业提供了以下优势: - **实时洞察:**实时数据湖允许企业访问最新的数据,从而做出更明智的决策。 - **数据民主化:**实时数据湖使各种利益相关者都可
recommend-type

spring添加xml配置文件

1. 创建一个新的Spring配置文件,例如"applicationContext.xml"。 2. 在文件头部添加XML命名空间和schema定义,如下所示: ``` <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。