并发服务器
服务器在同一个时刻可以响应多个客户端的请求
(1)多进程实现并发服务器
服务器端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <sys/wait.h>
#define SIZE 1024
#define PORT 9999
// sfd 用于和客户端通信
void handl_client(int sfd)
{
char buf[SIZE] = {0};
while (1)
{
int ret = read (sfd, buf, SIZE-1);
if (ret == -1)
{
perror ("read");
break;
}
if (ret == 0)
{
printf ("客户端退出\n");
break;
}
buf[ret] = '\0';
printf ("read %d bytes : %s\n", ret, buf);
int i;
for (i = 0; i < ret - 1; i++)
{
buf[i] = buf[i] + 'A' - 'a';
}
ret = write (sfd, buf, ret);
}
}
int init_tcp()
{
// 1、创建套接字
int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
if (listen_fd == -1)
{
perror ("创建套接字");
return -1;
}
struct sockaddr_in addr;
memset (&addr, 0, sizeof(addr));
addr.sin_family = AF_INET; // 使用IPv4地址族
addr.sin_port = htons(PORT); // 设置端口,要注意转换成网络字节序
addr.sin_addr.s_addr = htonl(INADDR_ANY); // 设置监听的IP地址,INADDR_ANY代表监听本机任意IP地址
// 2、绑定本地地址与端口(命令套接字)
int ret = bind(listen_fd, (struct sockaddr *)&addr, (socklen_t)sizeof(addr));
if (ret == -1)
{
perror ("绑定套接字");
return -1;
}
// 3、监听套接字、设置最大连接数
ret = listen(listen_fd, 5);
return listen_fd;
}
int myAccept(int listen_fd)
{
printf ("等待客户端的连接.....\n");
struct sockaddr_in client_addr;
int len = sizeof(client_addr);
int conn_fd = accept(listen_fd, (struct sockaddr *)&client_addr, (socklen_t *)&len);
if (conn_fd == -1)
{
perror ("accept");
return -1;
}
printf ("有一个客户端连接,地址是:%s\n", inet_ntoa(client_addr.sin_addr));
return conn_fd; // 将与客户端通信的 socket 返回
}
void handle(int sig)
{
int pid;
while (pid = waitpid(-1, NULL, WNOHANG) > 0)
{
printf ("检测到一个子进程 %d 退出,已成功收尸\n", pid);
}
}
int main()
{
// 初始化套接字
int listen_fd = init_tcp();
if (listen_fd == -1)
return -1;
signal(SIGCHLD, handle);
while (1)
{
// 等待客户端的连接
int conn_fd = myAccept(listen_fd);
if (conn_fd == -1)
{
continue;
}
// 创建一个子进程用来处理客户端的连接
pid_t pid = fork();
if (pid == 0) // 子进程,去处理客户端的请求
{
close (listen_fd); // 关闭连接套接字
handl_client(conn_fd); // 处理客户端的请求
close (listen_fd); // 处理完客户端后关闭与客户端的连接
exit(0); // 处理完客户端的请求后子进程退出
}
else if (pid > 0) // 父进程:监听客户端的连接,处理客户端连接,获取与客户端的连接描述符
{
close (conn_fd); // 父进程不与客户端进行通信,关闭与客户端的通信描述符
}
else
{
perror ("fork");
break;
}
}
close (listen_fd);
return 0;
}
客户端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define SIZE 1024
void ask_server(con_fd)
{
char buf[SIZE] = {0};
while (1)
{
fgets (buf, SIZE, stdin);
if (strncmp (buf, "end", 3) == 0)
{
break;
}
// 向服务器发送信息
int ret = write (con_fd, buf, strlen(buf));
if (ret == -1)
{
perror ("write");
break;
}
// 从服务器接收信息
ret = read (con_fd, buf, SIZE-1);
if (ret == -1)
{
perror ("read");
break;
}
buf[ret] = '\0';
printf ("从服务器接收到信息:%s\n", buf);
}
}
int main(int argc, char **argv)
{
if (argc != 3)
{
printf ("Usage: %s ip port\n", argv[0]);
return -1;
}
int port = atoi(argv[2]); // 获取端口
// 1、创建socket
int con_fd = socket(AF_INET, SOCK_STREAM,0);
if (con_fd == -1)
{
perror ("socket");
return -1;
}
struct sockaddr_in addr;
memset (&addr, 0, sizeof(addr));
addr.sin_family = AF_INET; // 使用IPv4地址族
addr.sin_port = htons(port); // 设置连接服务器的端口,要注意转换成网络字节序
addr.sin_addr.s_addr = inet_addr(argv[1]); // 设置连接的服务器的IP地址
// 2、连接服务器,第二个参数是连接的服务器的IP和端口
int ret = connect(con_fd, (struct sockaddr *)&addr, (socklen_t)sizeof(addr));
if (ret == -1)
{
perror ("connect");
return -1;
}
printf ("连接到服务器.....\n");
ask_server(con_fd);
close (con_fd);
return 0;
}
(2)多线程实现并发服务器,并进行简单的文件传输
服务器端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <pthread.h>
#include <sys/stat.h>
#include <fcntl.h>
#define SIZE 1024
#define PORT 9999
// sfd 用于和客户端通信
void* handl_client(void* v)
{
int sfd = (int)v;
int fd = open("1.ppt", O_RDONLY);
if (fd == -1)
{
perror ("open");
pthread_exit(NULL);
}
char buf[SIZE] = {0};
while (1)
{
int ret = read (fd, buf, SIZE-1);
if (ret == -1)
{
perror ("read");
break;
}
if (ret == 0)
{
printf ("文件传输结束\n");
break;
}
buf[ret] = '\0';
ret = write (sfd, buf, ret);
printf ("ret =%d\n", ret);
}
close (fd);
close (sfd);
}
int init_tcp()
{
// 1、创建套接字
int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
if (listen_fd == -1)
{
perror ("创建套接字");
return -1;
}
struct sockaddr_in addr;
memset (&addr, 0, sizeof(addr));
addr.sin_family = AF_INET; // 使用IPv4地址族
addr.sin_port = htons(PORT); // 设置端口,要注意转换成网络字节序
addr.sin_addr.s_addr = htonl(INADDR_ANY); // 设置监听的IP地址,INADDR_ANY代表监听本机任意IP地址
// 2、绑定本地地址与端口(命令套接字)
int ret = bind(listen_fd, (struct sockaddr *)&addr, (socklen_t)sizeof(addr));
if (ret == -1)
{
perror ("绑定套接字");
return -1;
}
// 3、监听套接字、设置最大连接数
ret = listen(listen_fd, 5);
return listen_fd;
}
int myAccept(int listen_fd)
{
printf ("等待客户端的连接.....\n");
struct sockaddr_in client_addr;
int len = sizeof(client_addr);
int conn_fd = accept(listen_fd, (struct sockaddr *)&client_addr, (socklen_t *)&len);
if (conn_fd == -1)
{
perror ("accept");
return -1;
}
printf ("有一个客户端连接,地址是:%s\n", inet_ntoa(client_addr.sin_addr));
return conn_fd; // 将与客户端通信的 socket 返回
}
int main()
{
// 初始化套接字
int listen_fd = init_tcp();
if (listen_fd == -1)
return -1;
while (1)
{
// 等待客户端的连接
int conn_fd = myAccept(listen_fd);
if (conn_fd == -1)
{
continue;
}
pthread_t thread;
// 创建线程来处理客户端的请求
int ret = pthread_create(&thread, NULL, handl_client, (void *)conn_fd);
if (ret == -1)
{
perror ("pthread_create");
break;
}
pthread_detach(thread); // 线程分离,线程结束系统回收资源
}
close (listen_fd);
return 0;
}
客户端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <sys/stat.h>
#include <fcntl.h>
#define SIZE 1024
void ask_server(con_fd)
{
int fd = open("2.ppt", O_WRONLY|O_CREAT, S_IRUSR|S_IWUSR);
if (fd == -1)
{
perror ("open");
pthread_exit(NULL);
}
char buf[SIZE] = {0};
while (1)
{
// 从服务器接收信息
int ret = read (con_fd, buf, SIZE-1);
if (ret == -1)
{
perror ("read");
break;
}
if (ret == 0)
{
printf ("接收文件成功\n");
return;
}
buf[ret] = '\0';
write(fd, buf, ret);
}
close (fd);
}
int main(int argc, char **argv)
{
if (argc != 3)
{
printf ("Usage: %s ip port\n", argv[0]);
return -1;
}
int port = atoi(argv[2]); // 获取端口
// 1、创建socket
int con_fd = socket(AF_INET, SOCK_STREAM,0);
if (con_fd == -1)
{
perror ("socket");
return -1;
}
struct sockaddr_in addr;
memset (&addr, 0, sizeof(addr));
addr.sin_family = AF_INET; // 使用IPv4地址族
addr.sin_port = htons(port); // 设置连接服务器的端口,要注意转换成网络字节序
addr.sin_addr.s_addr = inet_addr(argv[1]); // 设置连接的服务器的IP地址
// 2、连接服务器,第二个参数是连接的服务器的IP和端口
int ret = connect(con_fd, (struct sockaddr *)&addr, (socklen_t)sizeof(addr));
if (ret == -1)
{
perror ("connect");
return -1;
}
printf ("连接到服务器.....\n");
ask_server(con_fd);
close (con_fd);
return 0;
}
(3)多路复用select
服务器端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <sys/select.h>
#define SIZE 1024
#define PORT 9999
// sfd 用于和客户端通信
void handl_client(int sfd, fd_set *allset)
{
char buf[SIZE] = {0};
int ret = read (sfd, buf, SIZE-1);
if (ret == -1)
{
perror ("read");
return;
}
if (ret == 0)
{
printf ("客户端退出\n");
FD_CLR(sfd, allset);
return;
}
buf[ret] = '\0';
printf ("read %d bytes : %s\n", ret, buf);
int i;
for (i = 0; i < ret - 1; i++)
{
buf[i] = buf[i] + 'A' - 'a';
}
ret = write (sfd, buf, ret);
}
int init_tcp()
{
// 1、创建套接字
int listen_fd = socket(AF_INET, SOCK_STREAM, 0);
if (listen_fd == -1)
{
perror ("创建套接字");
return -1;
}
struct sockaddr_in addr;
memset (&addr, 0, sizeof(addr));
addr.sin_family = AF_INET; // 使用IPv4地址族
addr.sin_port = htons(PORT); // 设置端口,要注意转换成网络字节序
addr.sin_addr.s_addr = htonl(INADDR_ANY); // 设置监听的IP地址,INADDR_ANY代表监听本机任意IP地址
// 2、绑定本地地址与端口(命令套接字)
int ret = bind(listen_fd, (struct sockaddr *)&addr, (socklen_t)sizeof(addr));
if (ret == -1)
{
perror ("绑定套接字");
return -1;
}
// 3、监听套接字、设置最大连接数
ret = listen(listen_fd, 5);
return listen_fd;
}
int myAccept(int listen_fd)
{
printf ("等待客户端的连接.....\n");
struct sockaddr_in client_addr;
int len = sizeof(client_addr);
int conn_fd = accept(listen_fd, (struct sockaddr *)&client_addr, (socklen_t *)&len);
if (conn_fd == -1)
{
perror ("accept");
return -1;
}
printf ("有一个客户端连接,地址是:%s\n", inet_ntoa(client_addr.sin_addr));
return conn_fd; // 将与客户端通信的 socket 返回
}
int main()
{
// 初始化套接字
int listen_fd = init_tcp();
if (listen_fd == -1)
return -1;
int maxfd = listen_fd;
fd_set allset;
fd_set rset;
FD_ZERO(&allset); // 将文件描述符集清空
FD_SET(listen_fd, &allset); // 将监听文件描述符加入到监听序列
int client_fd[FD_SETSIZE]; // 代表select能够检测客户端数量
// 对数组进行初始化,全部置为-1,凡是值为-1的位置,都是空位置
int i;
for (i = 0; i < FD_SETSIZE; i++)
{
client_fd[i] = -1;
}
while (1)
{
rset = allset;
int count = select(maxfd + 1, &rset, NULL, NULL, NULL);
if (count == -1)
{
perror ("select");
break;
}
if (FD_ISSET(listen_fd, &rset)) // 查看是否有客户端要连接
{
int conn_fd = myAccept(listen_fd);
if (conn_fd == -1)
{
continue;
}
// 找个空位置存与客户端通信的文件描述符
for (i = 0; i < FD_SETSIZE; i++)
{
if (client_fd[i] == -1) // 找到一个空位置
{
client_fd[i] = conn_fd;
break;
}
}
// 判断文件描述符集是否已达上限
if (i == FD_SETSIZE)
{
close (conn_fd);
}
else
{
FD_SET(conn_fd, &allset);
if (maxfd < conn_fd)
maxfd = conn_fd;
}
if (--count <= 0)
continue;
}
for (i = 0; i < FD_SETSIZE; i++)
{
if (client_fd[i] == -1)
continue;
if (FD_ISSET(client_fd[i], &rset))
{
handl_client(client_fd[i], &allset);
if (--count <= 0)
break;
}
}
}
close (listen_fd);
return 0;
}
客户端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <string.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define SIZE 1024
void ask_server(con_fd)
{
char buf[SIZE] = {0};
while (1)
{
fgets (buf, SIZE, stdin);
if (strncmp (buf, "end", 3) == 0)
{
break;
}
// 向服务器发送信息
int ret = write (con_fd, buf, strlen(buf));
if (ret == -1)
{
perror ("write");
break;
}
// 从服务器接收信息
ret = read (con_fd, buf, SIZE-1);
if (ret == -1)
{
perror ("read");
break;
}
buf[ret] = '\0';
printf ("从服务器接收到信息:%s\n", buf);
}
}
int main(int argc, char **argv)
{
if (argc != 3)
{
printf ("Usage: %s ip port\n", argv[0]);
return -1;
}
int port = atoi(argv[2]); // 获取端口
// 1、创建socket
int con_fd = socket(AF_INET, SOCK_STREAM,0);
if (con_fd == -1)
{
perror ("socket");
return -1;
}
struct sockaddr_in addr;
memset (&addr, 0, sizeof(addr));
addr.sin_family = AF_INET; // 使用IPv4地址族
addr.sin_port = htons(port); // 设置连接服务器的端口,要注意转换成网络字节序
addr.sin_addr.s_addr = inet_addr(argv[1]); // 设置连接的服务器的IP地址
// 2、连接服务器,第二个参数是连接的服务器的IP和端口
int ret = connect(con_fd, (struct sockaddr *)&addr, (socklen_t)sizeof(addr));
if (ret == -1)
{
perror ("connect");
return -1;
}
printf ("连接到服务器.....\n");
ask_server(con_fd);
close (con_fd);
return 0;
}