linux网络编程之socket(四):使用fork并发处理多个client的请求和对等通信p2p

一、在前面讲过的回射客户/服务器程序中,服务器只能处理一个客户端的请求,如何同时服务多个客户端呢?在未讲到select/poll/epoll等高级IO之前,比较老土的办法是使用fork来实现。网络服务器通常用fork来同时服务多个客户端,父进程专门负责监听端口,每次accept一个新的客户端连接就fork出一个子进程专门服务这个客户端。但是子进程退出时会产生僵尸进程,父进程要注意处理SIGCHLD信号和调用wait清理僵尸进程,最简单的办法就是直接忽略SIGCHLD信号。

 C++ Code 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
/*************************************************************************
    > File Name: echoser.c
    > Author: Simba
    > Mail: [email protected]
    > Created Time: Fri 01 Mar 2013 06:15:27 PM CST
 ************************************************************************/


#include<stdio.h>
#include<sys/types.h>
#include<sys/socket.h>
#include<unistd.h>
#include<stdlib.h>
#include<errno.h>
#include<arpa/inet.h>
#include<netinet/in.h>
#include<string.h>
#include<signal.h>

#define ERR_EXIT(m) \
     do { \
        perror(m); \
        exit(EXIT_FAILURE); \
    }  while ( 0)

void do_service( int);

int main( void)
{
    signal(SIGCHLD, SIG_IGN);
     int listenfd;  //被动套接字(文件描述符),即只可以accept, 监听套接字
     if ((listenfd = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP)) <  0)
         //  listenfd = socket(AF_INET, SOCK_STREAM, 0)
        ERR_EXIT( "socket error");

     struct sockaddr_in servaddr;
    memset(&servaddr,  0sizeof(servaddr));
    servaddr.sin_family = AF_IN
  • 14
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
可以使用fork()函数创建多个进程,每个进程都可以通过socket进行通信。以下是一个简单的示例代码: Server端示例代码: ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <arpa/inet.h> #define PORT 8888 #define MAX_CONN 5 int main() { int server_fd, new_socket, valread; struct sockaddr_in address; int opt = 1; int addrlen = sizeof(address); char buffer[1024] = {}; char *hello = "Hello from server"; // 创建socket if ((server_fd = socket(AF_INET, SOCK_STREAM, )) == ) { perror("socket failed"); exit(EXIT_FAILURE); } // 设置socket选项 if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt))) { perror("setsockopt"); exit(EXIT_FAILURE); } address.sin_family = AF_INET; address.sin_addr.s_addr = INADDR_ANY; address.sin_port = htons(PORT); // 绑定socket if (bind(server_fd, (struct sockaddr *)&address, sizeof(address)) < ) { perror("bind failed"); exit(EXIT_FAILURE); } // 监听socket if (listen(server_fd, MAX_CONN) < ) { perror("listen"); exit(EXIT_FAILURE); } // 接受连接请求 while (1) { if ((new_socket = accept(server_fd, (struct sockaddr *)&address, (socklen_t*)&addrlen)) < ) { perror("accept"); exit(EXIT_FAILURE); } // 创建子进程处理连接 if (fork() == ) { close(server_fd); // 读取客户端发送的数据 valread = read(new_socket, buffer, 1024); printf("%s\n", buffer); // 发送数据给客户端 send(new_socket, hello, strlen(hello), ); printf("Hello message sent\n"); close(new_socket); exit(); } close(new_socket); } return ; } ``` Client端示例代码: ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <arpa/inet.h> #define PORT 8888 int main(int argc, char const *argv[]) { int sock = , valread; struct sockaddr_in serv_addr; char *hello = "Hello from client"; char buffer[1024] = {}; // 创建socket if ((sock = socket(AF_INET, SOCK_STREAM, )) < ) { printf("\n Socket creation error \n"); return -1; } serv_addr.sin_family = AF_INET; serv_addr.sin_port = htons(PORT); // 将IP地址从点分十进制转换为二进制格式 if(inet_pton(AF_INET, "127...1", &serv_addr.sin_addr)<=) { printf("\nInvalid address/ Address not supported \n"); return -1; } // 连接服务器 if (connect(sock, (struct sockaddr *)&serv_addr, sizeof(serv_addr)) < ) { printf("\nConnection Failed \n"); return -1; } // 发送数据给服务器 send(sock, hello, strlen(hello), ); printf("Hello message sent\n"); // 读取服务器发送的数据 valread = read(sock, buffer, 1024); printf("%s\n", buffer); return ; } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值