📄 server.c
字号:
/********************************************************** * server.c -- 基于TCP协议的服务器方演示程序 * 运行在2410实验箱端(请要用交叉编译器编译该程序) * 其中一些注释的语句也是可以用的 * 功能描述: * 服务器首先创建一个socket,然后将socket与本地地址/端口捆绑, * 成功之后就在相应的socket上监听.当accept()捕捉到一个 * 连接服务请求时就生成一个新的socket,并通过这个新的socket向客户 * 发送字符串"Hello World!",然后关闭该socket **********************************************************/#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <errno.h>#include <string.h>#include <sys/types.h>#include <sys/socket.h>#include <netinet/in.h>#include <arpa/inet.h>#include <sys/wait.h>#include <signal.h>#define MYPORT 3490 // 用户端要监听的端口号#define BACKLOG 10 // 多少个队列被挂起void sigchld_handler(int s){ while(wait(NULL) > 0);}int main(void){ int sockfd, new_fd; // 在sockfd上监听;在 new_fd 上进行新的连接 struct sockaddr_in my_addr; //本机的地址信息 struct sockaddr_in their_addr; //连接方的地址信息 int sin_size; struct sigaction sa; int yes=1; if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) { perror("socket"); exit(1); } if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int)) == -1) { perror("setsockopt"); exit(1); } my_addr.sin_family = AF_INET; // 地址族,一般为AF_INET my_addr.sin_port = htons(MYPORT); // 端口号,主机地址字节顺序转向网络字节顺序(对短型数据操作) //my_addr.sin_addr.s_addr = INADDR_ANY; // 填入本机IP地址 my_addr.sin_addr.s_addr = inet_addr("192.168.1.45"); // 填入本机IP地址192.168.1.45 //my_addr.sin_addr.s_addr = htonl(0xc0a8012d); // 填入本机IP地址 //my_addr.sin_addr.s_addr = 0x2d01a8c0U; // 也可以直接填入本机IP地址,但字节顺序是反的 memset(&(my_addr.sin_zero), '\0', 8); // 填充0以保持与结构体sockaddr同样大小 if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) == -1) { //将套接字与IP地址192.168.1.45绑定 perror("bind"); exit(1); } if (listen(sockfd, BACKLOG) == -1) { //监听是否有服务请求 perror("listen"); exit(1); } sa.sa_handler = sigchld_handler; // 回收所有的死进程 sigemptyset(&sa.sa_mask); sa.sa_flags = SA_RESTART; if (sigaction(SIGCHLD, &sa, NULL) == -1) { perror("sigaction"); exit(1); } while(1) // 主 accept() 回路 { sin_size = sizeof(struct sockaddr_in); if ((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, &sin_size)) == -1) { //参数1:socket文件描述符;参数2:客户机端的连接信息;参数3:参数2的长度. //当某个客户端试图与服务器监听的端口连接时,该连接请求将排队等待服务器调用accept()接受它. //程序调用accept()为该请求建立一个连接,并返回一个新的socket描述符new_fd,供该新连接使用 perror("accept"); continue; } printf("server: got connection from %s\n",inet_ntoa(their_addr.sin_addr)); if (!fork()) { // 创建一个子进程 close(sockfd); // 子进程不需要监听,但在主进程中监听仍然有效 if (send(new_fd, "Hello, world!\n", 14, 0) == -1) perror("send"); close(new_fd); exit(0); } close(new_fd); // 父进程不需要accept()的连接,因而要关闭它 } return 0;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -