echo服务端-单线程
- 一次只能处理一个请求
- 可以开启第二个客户端观察,会发现请求堵塞,但只要停掉第一个客户端,立马就能返回结果
echo_server.c
#include <stdio.h> //printf、fput
#include <stdlib.h> //atoi
#include <string.h> //memset
#include <unistd.h> //write、close
#include <arpa/inet.h> //sockaddr_in结构、INADDR_ANY
#include <sys/socket.h>
#define BUF_SIZE 100
int main(int argc, char *argv[]) {
char buf[BUF_SIZE];
if (argc != 2) {
printf("Usage: %s <port>\n", argv[0]);
exit(1);
}
int serv_sock = socket(PF_INET, SOCK_STREAM, 0);
if (serv_sock == -1) {
fputs("socket() error", stderr);
exit(1);
}
struct sockaddr_in serv_addr;
memset(&serv_addr, 0, sizeof(serv_addr)); //全部位置0
serv_addr.sin_family = AF_INET; //
serv_addr.sin_addr.s_addr = htonl(INADDR_ANY); //等价于绑定到"0.0.0.0"
serv_addr.sin_port = htons(atoi(argv[1])); //取1号参数作为端口
if (bind(serv_sock, (struct sockaddr*) &serv_addr, sizeof(serv_addr)) == -1) {
fputs("bind() error", stderr);
exit(1);
}
if (listen(serv_sock, 5) == -1) {
fputs("listen() error", stderr);
exit(1);
}
while(1) {
struct sockaddr_in clnt_addr;
socklen_t clnt_addr_size = sizeof(clnt_addr);
int clnt_sock = accept(serv_sock, (struct sockaddr*) &clnt_addr, &clnt_addr_size);
if (clnt_sock == -1) {
fputs("accept() error", stderr);
exit(1);
}
while(1) {
int str_len = read(clnt_sock, buf, BUF_SIZE);
if (str_len == 0) {
close(clnt_sock);
printf("close client sock: %d \n", clnt_sock);
break;
} else {
write(clnt_sock, buf, str_len); //echo
}
}
}
close(serv_sock);
return 0;
}
-
编译
gcc echo_server.c -o eserver -
执行
./eserver 8100