#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <arpa/inet.h> #include <sys/socket.h> #define BUF_SIZE 1024 #define OPSZ 4 void error_handling(char message); int calculate(int opnum, int opnds[], char oprator); int main(int argc, char argv[]) { int serv_sock, clnt_sock; char opinfo[BUF_SIZE]; int result, opnd_cnt, i; int recv_cnt, recv_len; struct sockaddr_in serv_adr, clnt_adr; socklen_t clnt_adr_sz; if(argc!=2) { printf("Usage : %s <port>\n", argv[0]); exit(1); } serv_sock=socket(PF_INET, SOCK_STREAM, 0); if(serv_sock==-1) error_handling("socket() error"); memset(&serv_adr, 0, sizeof(serv_adr)); serv_adr.sin_family=AF_INET; serv_adr.sin_addr.s_addr=htonl(INADDR_ANY); serv_adr.sin_port=htons(atoi(argv[1])); if(bind(serv_sock, (struct sockaddr)&serv_adr, sizeof(serv_adr))==-1) error_handling("bind() error"); if(listen(serv_sock, 5)==-1) error_handling("listen() error"); clnt_adr_sz=sizeof(clnt_adr); for(i=0; i<5; i++) { opnd_cnt=0; clnt_sock=accept(serv_sock, (struct sockaddr)&clnt_adr, &clnt_adr_sz); read(clnt_sock, &opnd_cnt, 1); recv_len=0; while((opnd_cntOPSZ+1)>recv_len) { recv_cnt=read(clnt_sock, &opinfo[recv_len], BUF_SIZE-1); recv_len+=recv_cnt; } result=calculate(opnd_cnt, (int)opinfo, opinfo[recv_len-1]); write(clnt_sock, (char*)&result, sizeof(result)); close(clnt_sock); } close(serv_sock); return 0; } int calculate(int opnum, int opnds[], char op) { int result=opnds[0], i; switch(op) { case '+': for(i=1; i<opnum; i++) result+=opnds[i]; break; case '-': for(i=1; i<opnum; i++) result-=opnds[i]; break; case '': for(i=1; i<opnum; i++) result=opnds[i]; break; } return result; } void error_handling(char *message) { fputs(message, stderr); fputc('\n', stderr); exit(1); } 对每行代码给出详细解释
时间: 2024-03-20 18:44:32 浏览: 141
用c语言实现的逆序文本程序
这段代码是一个简单的服务端程序,用于接收客户端传来的数学表达式并返回计算结果。以下是代码的详细解释:
```c
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#define BUF_SIZE 1024 // 定义读取缓冲区大小
#define OPSZ 4 // 定义操作数大小
void error_handling(char *message); // 定义错误处理函数
int calculate(int opnum, int opnds[], char oprator); // 定义计算函数
int main(int argc, char *argv[]) { // 主函数
int serv_sock, clnt_sock; // 定义服务端和客户端socket
char opinfo[BUF_SIZE]; // 定义读取缓冲区
int result, opnd_cnt, i; // 定义计算结果、操作数个数、循环计数器
int recv_cnt, recv_len; // 定义接收字节数、接收字节长度
struct sockaddr_in serv_adr, clnt_adr; // 定义服务端和客户端地址
socklen_t clnt_adr_sz; // 定义客户端地址长度
if(argc!=2) { // 判断参数是否正确
printf("Usage : %s <port>\n", argv[0]);
exit(1);
}
serv_sock=socket(PF_INET, SOCK_STREAM, 0); // 创建socket
if(serv_sock==-1)
error_handling("socket() error");
memset(&serv_adr, 0, sizeof(serv_adr)); // 清空服务端地址
serv_adr.sin_family=AF_INET; // 指定地址族为IPv4
serv_adr.sin_addr.s_addr=htonl(INADDR_ANY); // 指定IP地址为本地任意地址
serv_adr.sin_port=htons(atoi(argv[1])); // 指定端口号
if(bind(serv_sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr))==-1) // 绑定地址
error_handling("bind() error");
if(listen(serv_sock, 5)==-1) // 监听端口
error_handling("listen() error");
clnt_adr_sz=sizeof(clnt_adr); // 计算客户端地址长度
for(i=0; i<5; i++) { // 处理5个客户端请求
opnd_cnt=0; // 操作数个数初始化
clnt_sock=accept(serv_sock, (struct sockaddr*)&clnt_adr, &clnt_adr_sz); // 接受客户端请求
read(clnt_sock, &opnd_cnt, 1); // 读取操作数个数
recv_len=0; // 接收长度初始化
while((opnd_cnt*OPSZ+1)>recv_len) { // 循环读取操作数和操作符
recv_cnt=read(clnt_sock, &opinfo[recv_len], BUF_SIZE-1); // 读取操作数和操作符
recv_len+=recv_cnt; // 更新接收长度
}
result=calculate(opnd_cnt, (int*)opinfo, opinfo[recv_len-1]); // 计算结果
write(clnt_sock, (char*)&result, sizeof(result)); // 发送计算结果
close(clnt_sock); // 关闭客户端socket
}
close(serv_sock); // 关闭服务端socket
return 0;
}
int calculate(int opnum, int opnds[], char op) { // 计算函数
int result=opnds[0], i; // 初始化计算结果为第一个操作数
switch(op) { // 根据操作符计算结果
case '+':
for(i=1; i<opnum; i++)
result+=opnds[i];
break;
case '-':
for(i=1; i<opnum; i++)
result-=opnds[i];
break;
case '*':
for(i=1; i<opnum; i++)
result*=opnds[i];
break;
case '/':
for(i=1; i<opnum; i++)
result/=opnds[i];
break;
}
return result; // 返回计算结果
}
void error_handling(char *message) { // 错误处理函数
fputs(message, stderr); // 输出错误信息
fputc('\n', stderr); // 输出换行符
exit(1); // 退出程序
}
```
代码主要分为以下几部分:
1. 定义常量和函数。
2. 主函数部分,包括socket创建、地址绑定、监听端口、处理客户端请求等。
3. 计算函数部分,根据操作数和操作符计算表达式的结果。
4. 错误处理函数部分,输出错误信息并退出程序。
阅读全文