说明

在Linux中通过流式套接字编程(TCP),实现一个并发服务器的访问回显,适合刚学完Linux套接字编程的朋友进行巩固训练
具体功能:

  • 服务器能够同时连接、处理多个客户端的信息
  • 客户端向服务器发送数据之后,服务器收到数据,然后反手发送给客户端
  • 服务器能够对客户端的退出做出反应,并在客户端退出连接的时候给出提示
  • 服务器能够识别每个客户端发送的信息,在显示的时候加上客户端的IP地址
  • 服务器中能够对已经退出的服务进程作回收处理
  • 客户端能够对服务器的退出作出反应,检测到服务器退出后客户端也退出

注意事项

  • 多进程并发服务器编程中,每次建立一个套接字连接,都会fork一个进程来处理
  • accept是自带阻塞的,所以fork返回父进程之后,父进程就会阻塞等待下一个已连接套接字
  • 客户端的关闭通过ctrl-c发出的信号(SIGINT)来终止客户端
  • 当客户端终止之后,服务器上对应的服务进程通过exit结束,此时由于服务器的主进程还阻塞在accept中,所以无法及时回收子服务进程,所以通过注册一个信号SIGCHLD处理函数,在信号处理的时候回收僵尸子进程。SIGCHLD是子进程结束的时候发送给父进程的信号,默认忽略。
  • 服务器进程如何检测客户端退出呢?通过recv()函数,当返回值为0的时候,表示客户端已经关闭套接字,即客户端退出。
  • 当服务线程主动关闭的时候,客户端也会通过recv()收到服务器关闭的信息,然后客户端主动退出
  • 关于套接字描述符,因为描述符也算是进程的资源,当套接字描述符的引用值为0的时候,才会关闭套接字,或者是进程退出的时候释放套接字描述符资源
  • 每次fork的时候,都会产生一个对于已经打开的套接字描述符的引用,所以要在进入子服务进程后关闭监听套接字描述符、在主服务进程中关闭已连接套接字描述符、在子服务进程退出的时候关闭已连接套接字描述符、在退出主服务器进程的时候关闭监听套接字描述符,这样才做到了有始有终(fork之后已连接套接字描述符的引用就有两份)

server.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <signal.h>
#include <errno.h> #define SERVER_ADDR "172.17.44.154"
#define BUFSIZE 100 void sigchld_handler(int arg); int main(int argc, const char *argv[])
{
int socket_fd, new_fd;
struct sockaddr_in server_addr, cli_addr;
char buf[BUFSIZE];
int pid;
struct sigaction sig; /* 注册中断信号处理函数 */
sig.sa_handler = sigchld_handler;
sigaction(SIGCHLD, &sig, NULL); /* 创建套接字,并获取套接字描述符 */
socket_fd = socket(AF_INET, SOCK_STREAM, 0);
if (-1 == socket_fd) {
perror("socket");
exit(-1);
} /* 绑定地址 */
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(5001);
inet_pton(AF_INET, SERVER_ADDR, (void*)&server_addr.sin_addr.s_addr); //地址转换
if (-1 == bind(socket_fd, (struct sockaddr*)&server_addr, sizeof(server_addr))) {
perror("bind");
exit(-1);
} /* 转换为被动连接套接字 */
if (-1 == listen(socket_fd, 5)) {
perror("listen");
exit(-1);
} #if 0 //单进程服务器
/* 获取已连接套接字 */
socklen_t len = 0;
new_fd = accept(socket_fd, (struct sockaddr*)&cli_addr, (socklen_t *)&len);
if (-1 == new_fd) {
perror("accept");
exit(-1);
}
printf("accept socket!\nclient ip :%s port:%d\n", inet_ntoa(cli_addr.sin_addr), cli_addr.sin_port); while (1) {
memset(buf, 0, BUFSIZE);
if (0 == recv(new_fd, buf, BUFSIZE, 0)) { //接受数据
printf("the client is closed\n");
break;
}
printf("read:%s\n", buf);
send(new_fd, buf, BUFSIZE, 0); //回应客户端
}
close(new_fd); #else //多进程并发服务器
while (1) { socklen_t len = 0;
new_fd = accept(socket_fd, (struct sockaddr*)&cli_addr, (socklen_t *)&len);
if (-1 == new_fd) {
if (errno == EINTR) continue; //accept可能会被信号中断
perror("accept");
exit(-1);
} /* 并发服务器:子进程中进行TCP通信 */
pid = fork();
if (pid == -1) {
perror("fork");
exit(0);
} else if (pid == 0) {
close(socket_fd); //关闭监听套接字 while (1) {
memset(buf, 0, BUFSIZE);
if (0 == recv(new_fd, buf, BUFSIZE, 0)) { //接受数据,只有当客户端主动关闭的时候,才退出线程,还要对关闭之后的子进程
printf("the client socket %s is closed\n", inet_ntoa((struct in_addr)cli_addr.sin_addr));
close(new_fd); //退出之前记得关闭 已连接套接字
exit(0); //通过信号处理函数进行回收 }
getsockname(new_fd, (struct sockaddr*)&cli_addr, (socklen_t *)&len); //获取连接套接字信息
printf("recv client IP:%s data:%s\n", inet_ntoa((struct in_addr)cli_addr.sin_addr), buf);
send(new_fd, buf, BUFSIZE, 0); //回应客户端
} } else {
close(new_fd); //父进程关闭 已连接套接字
} } #endif close(socket_fd);
return 0;
} void sigchld_handler(int arg)
{
int child_pid;
if (SIGCHLD == arg) {
if ((child_pid = waitpid(-1, NULL, WNOHANG)) == -1) {
perror("sigchld");
}
printf("a client %d is end\n", child_pid);
} }

client.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#define SERVER_ADDR "172.17.44.154"
#define BUFSIZE 100 int main(int argc, const char *argv[])
{
int new_fd;
struct sockaddr_in server_addr;
char buf[BUFSIZE]; /* 创建套接字,并获取套接字描述符 */
new_fd = socket(AF_INET, SOCK_STREAM, 0);
if (-1 == new_fd) {
perror("socket");
exit(-1);
} server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(5001);
inet_pton(AF_INET, SERVER_ADDR, (void*)&server_addr.sin_addr.s_addr);
if (-1 == connect(new_fd, (struct sockaddr*)&server_addr, sizeof(server_addr))) {
perror("connect");
exit(-1);
} while (1) {
printf("input:");
fgets(buf, BUFSIZE, stdin); //获取数据 if (-1 == send(new_fd, buf, BUFSIZE, 0)) { //发送数据
perror("send");
close(new_fd);
exit(-1);
}
if (0 == recv(new_fd, buf, BUFSIZE, 0)) { //收到数据
printf("server closed\n");
break;
}
printf("recv:%s\n", buf);
}
close(new_fd); return 0;
}

运行截图

PS:这里是在同一主机下做实验的,所以各个客户端的IP地址都是一样的
正常运行的状态如下:

当有一个客户端退出时,服务器会显示信息,但是对其他客户端的服务正常进行:

当服务器主动关闭之后,所有客户端都会收到服务器关闭的信息,并且主动退出:

Linux 并发服务器编程(多进程)的更多相关文章

  1. Linux 高性能服务器编程——多进程编程

    问题聚焦:     进程是Linux操作系统环境的基础.     本篇讨论以下几个内容,同时也是面试经常被问到的一些问题:     1 复制进程映像的fork系统调用和替换进程映像的exec系列系统调 ...

  2. Linux 高性能服务器编程——高性能服务器程序框架

    问题聚焦:     核心章节.     服务器一般分为如下三个主要模块:I/O处理单元(四种I/O模型,两种高效事件处理模块),逻辑单元(两种高效并发模式,有效状态机)和存储单元(不讨论). 服务器模 ...

  3. Linux 高性能服务器编程——Linux网络编程基础API

    问题聚焦:     这节介绍的不仅是网络编程的几个API     更重要的是,探讨了Linux网络编程基础API与内核中TCP/IP协议族之间的关系.     这节主要介绍三个方面的内容:套接字(so ...

  4. 第15章 高并发服务器编程(2)_I/O多路复用

    3. I/O多路复用:select函数 3.1 I/O多路复用简介 (1)通信领域的时分多路复用 (2)I/O多路复用(I/O multiplexing) ①同一线程,通过“拨开关”方式,来同时处理多 ...

  5. linux高性能服务器编程

    <Linux高性能服务器编程>:当当网.亚马逊 目录: 第一章:tcp/ip协议族 第二章:ip协议族 第三章:tcp协议详解 第四章:tcp/ip通信案例:访问Internet 第五章: ...

  6. linux高性能服务器编程 (一) --Tcp/Ip协议族

    前言: 在学习swoole入门基础的过程中,遇到了很多知识瓶颈,比方说多进程.多线程.以及进程池和线程池等都有诸多的疑惑.之前也有学习相关知识,但只是单纯的知识面了解.而没有真正的学习他们的来龙去脉. ...

  7. Linux 高性能服务器编程——多线程编程

    问题聚焦:     在简单地介绍线程的基本知识之后,主要讨论三个方面的内容:    1 创建线程和结束线程:    2 读取和设置线程属性:    3 线程同步方式:POSIX信号量,互斥锁和条件变量 ...

  8. Linux 高性能服务器编程——I/O复用

    问题聚焦:     前篇提到了I/O处理单元的四种I/O模型.     本篇详细介绍实现这些I/O模型所用到的相关技术.     核心思想:I/O复用 使用情景: 客户端程序要同时处理多个socket ...

  9. Linux 高性能服务器编程——Linux服务器程序规范

    问题聚焦:     除了网络通信外,服务器程序通常还必须考虑许多其他细节问题,这些细节问题涉及面逛且零碎,而且基本上是模板式的,所以称之为服务器程序规范.     工欲善其事,必先利其器,这篇主要来探 ...

随机推荐

  1. 使用 redis 减少 秒杀库存 超卖思路 (转)

      由于数据库查询的及插入的操作 耗费的实际时间要耗费比redis 要多, 导致 多人查询时库存有,但是实际插入数据库时却超卖 redis 会有效的减少相关的延时,对于并发量相对较少的 可以一用 1 ...

  2. buu 相册

    一.拖入jeb,这个神器里面,感觉对jeb使用还是不熟悉,对我逆向产生了一些障碍. 抓住题目给的提示,邮箱,全局直接搜索,mail. 看下它的交叉引用 找到了发邮件的方法, C2的MAILFROME说 ...

  3. EF Core3.1 CodeFirst动态自动添加表和字段的描述信息

    前言 我又来啦.. 本篇主要记录如何针对CodeFirst做自动添加描述的扩展 为什么要用这个呢.. 因为EF Core3.1 CodeFirst 对于自动添加描述这块 只有少部分的数据库支持.. 然 ...

  4. Linux基本操作 [转]

    前言 只有光头才能变强 这个学期开了Linux的课程了,授课的老师也是比较负责任的一位.总的来说也算是比较系统地学习了一下Linux了~~~ 本文章主要是总结Linux的基础操作以及一些简单的概念~如 ...

  5. Vue高阶

    Vue.cli是基于vue应用开发提供的一个脚手架工具,为应用搭建基础的框架架构,提供插件.开发服务.打包等功能. 1. 安装 node.js是一个JavaScript的运行环境,提供了一个事件驱动. ...

  6. NSIS 插件开发引发的思考

    支持NSIS的DLL扩展编程通用语法结构 #include <windows.h> #include <stdio.h> #define FORCE_SWITCH " ...

  7. Python单元测试框架unittest之生成测试报告(HTMLTestRunner)

    前言 批量执行完用例后,生成的测试报告是文本形式的,不够直观,为了更好的展示测试报告,最好是生成HTML格式的. unittest里面是不能生成html格式报告的,需要导入一个第三方的模块:HTMLT ...

  8. 「CF526F」 Pudding Monsters

    CF526F Pudding Monsters 传送门 模型转换:对于一个 \(n\times n\) 的棋盘,若每行每列仅有一个棋子,令 \(a_x=y\),则 \(a\) 为一个排列. 转换成排列 ...

  9. python从图片中找图

    import aircv as ac def matcha(bb,aa):#从bb查找aa,如果有则返回其坐标位置 yuan=ac.imread(bb) mubi=ac.imread(aa) resu ...

  10. HTML表单__表单元素属性

    看完"HTML表单__表单元素"那一节的同学会发现,同是input标签,type属性值不一样的时候,input类型完全不一样.type就是input的一个属性,除type之外,还有 ...