日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

step5 . day6 网络编程 基于TCP协议的多并发模式(使用多进程、多线程、select函数分别实现)...

發(fā)布時間:2024/4/17 编程问答 26 豆豆
生活随笔 收集整理的這篇文章主要介紹了 step5 . day6 网络编程 基于TCP协议的多并发模式(使用多进程、多线程、select函数分别实现)... 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

實(shí)現(xiàn)TCP服務(wù)器端多路并發(fā)的方法有①多進(jìn)程②多線程③IO多路復(fù)用(select poll epoll函數(shù))

1.多進(jìn)程實(shí)現(xiàn)并發(fā)模式(僅在服務(wù)器端更改之前代碼實(shí)現(xiàn))

服務(wù)器端
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <linux/in.h>
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <signal.h>
#include <sys/wait.h>

#define SIZE 128

void handler(int sign);
int main(int argc, const char *argv[])
{
if(argc<2){
printf("Usrmsg:%s <Port>\n",argv[0]);
return -1;
}
//1.創(chuàng)建套接字文件,用于鏈接
int socket_fd;
socket_fd = socket(AF_INET,SOCK_STREAM,0);
if(socket_fd == -1){
perror("socket");
return -1;
}
printf("socket Created ok\n");

//填充結(jié)構(gòu)體sockaddr_in,用于傳參
int bind_fd;
struct sockaddr_in serveraddr,connectaddr; //定義填充的結(jié)構(gòu)體
bzero((void *)&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET; //IPV4協(xié)議
serveraddr.sin_port = htons(atoi(argv[1])); //本地端口號轉(zhuǎn)化位網(wǎng)絡(luò)字節(jié)碼
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);//本地IP轉(zhuǎn)化位網(wǎng)絡(luò)字節(jié)碼,任意IP
//2.綁定socket文件描述符、端口、IP
bind_fd = bind(socket_fd,(struct sockaddr*)&serveraddr,sizeof(serveraddr));
if(bind_fd == -1){
perror("bind");
return -1;
}
printf("bind ok\n");

//3.監(jiān)聽,將套接字文件屬性該主動為被動模式
int listen_fd;
listen_fd = listen(socket_fd,10);

if(listen_fd == -1){
perror("listen");
return -1;
}
printf("listen ok\n");


int addrsize = sizeof(connectaddr);
int accept_fd;
int recvbyte;

char bufrecv[SIZE] = {0};
while(1){

accept_fd = accept(socket_fd,(struct sockaddr*)&connectaddr,&addrsize);
if(accept_fd == -1){
perror("accept");
return -1;
}

printf("connect ok\n");
pid_t pid = fork();

if(pid < 0){
perror("fork");
return -1;
}

if(pid == 0){
close(socket_fd);
while(1){
memset(bufrecv,0,SIZE);
int port = ntohs(connectaddr.sin_port);
char *ip = (char *)(inet_ntoa(connectaddr.sin_addr.s_addr));

recvbyte = recv(accept_fd, bufrecv,sizeof(bufrecv),0);

if(recvbyte <= 0){
printf("client failed connect\n");
return -1;
}
printf("IP:%s Port:%d online. send: %s\n",ip,port,bufrecv);
}
// close(accept_fd);
}
else{
close(accept_fd);
signal(SIGCHLD,handler);

}
}

close(socket_fd);

return 0;
}
void handler(int sign){

waitpid(-1,NULL,WNOHANG);
printf("signal = %d\n",sign);
}

?

2.多線程實(shí)現(xiàn)并發(fā)


#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <linux/in.h>
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <pthread.h>

#define SIZE 128

void *connectfun(void *arg);
int main(int argc, const char *argv[])
{
if(argc<2){
printf("Usrmsg:%s <Port>\n",argv[0]);
return -1;
}
//1.創(chuàng)建套接字文件,用于鏈接
int socket_fd;
socket_fd = socket(AF_INET,SOCK_STREAM,0);
if(socket_fd == -1){
perror("socket");
return -1;
}
printf("socket Created ok\n");

//填充結(jié)構(gòu)體sockaddr_in,用于傳參
int bind_fd;
struct sockaddr_in serveraddr,connectaddr; //定義填充的結(jié)構(gòu)體
bzero((void *)&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET; //IPV4協(xié)議
serveraddr.sin_port = htons(atoi(argv[1])); //本地端口號轉(zhuǎn)化位網(wǎng)絡(luò)字節(jié)碼
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);//本地IP轉(zhuǎn)化位網(wǎng)絡(luò)字節(jié)碼,任意IP
//2.綁定socket文件描述符、端口、IP
bind_fd = bind(socket_fd,(struct sockaddr*)&serveraddr,sizeof(serveraddr));
if(bind_fd == -1){
perror("bind");
return -1;
}
printf("bind ok\n");

//3.監(jiān)聽,將套接字文件屬性該主動為被動模式
int listen_fd;
listen_fd = listen(socket_fd,10);

if(listen_fd == -1){
perror("listen");
return -1;
}
printf("listen ok\n");


int addrsize = sizeof(connectaddr);
int accept_fd;

while(1){

accept_fd = accept(socket_fd,(struct sockaddr*)&connectaddr,&addrsize);
if(accept_fd == -1){
perror("accept");
return -1;
}
printf("connect ok\n");

pthread_t pid ;
pthread_create(&pid, NULL,connectfun,(void*)&accept_fd);

}

close(socket_fd);
return 0;

}
void *connectfun(void *arg){
pthread_detach(pthread_self());
char bufrecv[SIZE] = {0};
int recvbyte;
while(1){

memset(bufrecv,0,SIZE);

recvbyte = recv(*(int *)arg, bufrecv,sizeof(bufrecv),0);

if(recvbyte <= 0){
printf("client failed connect\n");
return NULL;
}
printf("send: %s\n",bufrecv);
}
}

?

3.select函數(shù)實(shí)現(xiàn)多并發(fā)


#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <linux/in.h>
#include <string.h>
#include <sys/stat.h>
#include <fcntl.h>

#define SIZE 128

int main(int argc, const char *argv[])
{
if(argc<2){
printf("Usrmsg:%s <Port>\n",argv[0]);
return -1;
}
//1.創(chuàng)建套接字文件,用于鏈接
int socket_fd;
socket_fd = socket(AF_INET,SOCK_STREAM,0);
if(socket_fd == -1){
perror("socket");
return -1;
}
printf("socket Created ok\n");

//填充結(jié)構(gòu)體sockaddr_in,用于傳參
int bind_fd;
struct sockaddr_in serveraddr,connectaddr; //定義填充的結(jié)構(gòu)體
bzero((void *)&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET; //IPV4協(xié)議
serveraddr.sin_port = htons(atoi(argv[1])); //本地端口號轉(zhuǎn)化位網(wǎng)絡(luò)字節(jié)碼
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);//本地IP轉(zhuǎn)化位網(wǎng)絡(luò)字節(jié)碼,任意IP
//2.綁定socket文件描述符、端口、IP
bind_fd = bind(socket_fd,(struct sockaddr*)&serveraddr,sizeof(serveraddr));
if(bind_fd == -1){
perror("bind");
return -1;
}
printf("bind ok\n");

//3.監(jiān)聽,將套接字文件屬性該主動為被動模式
int listen_fd;
listen_fd = listen(socket_fd,10);

if(listen_fd == -1){
perror("listen");
return -1;
}
printf("listen ok\n");

//創(chuàng)建監(jiān)聽集合
fd_set readfds,tempfds;
//清空集合
FD_ZERO(&readfds);
FD_ZERO(&tempfds);
//將關(guān)心的文件描述符加入集合
FD_SET(0,&readfds);
FD_SET(socket_fd,&readfds);

//設(shè)置最大監(jiān)聽個數(shù)
int maxfd = socket_fd;

int addrsize = sizeof(connectaddr);
int accept_fd;
int recvbyte;
int val;
char bufsend[SIZE] = {0};
char bufrecv[SIZE] = {0};
while(1){

tempfds = readfds;
val = select(maxfd + 1,&tempfds,NULL,NULL,NULL);
//4.阻塞函數(shù),等待客戶鏈接請求,不關(guān)心鏈接的是那個客戶端,填NULL
if(val < 0)
{
perror("select failed.");
return -1;
}
int i = 0;
for(i = 0;i<maxfd+1;i++){
if(FD_ISSET(i,&tempfds)){

if(i == 0)
{
fgets(bufsend,sizeof(bufsend),stdin);

send(socket_fd, bufsend, sizeof(bufsend),0);
printf("send ok.\n");
}
if(i == socket_fd)
{
accept_fd = accept(socket_fd,(struct sockaddr*)&connectaddr,&addrsize);
if(accept_fd == -1){
perror("accept");
return -1;
}

printf("accept ok\n");
int port = ntohs(connectaddr.sin_port);
char *ip = (char *)(inet_ntoa(connectaddr.sin_addr.s_addr));
printf("IP:%s Port:%d online.\n",ip,port);
FD_SET(accept_fd,&readfds);
maxfd = maxfd > accept_fd ? maxfd : accept_fd;

}
else{
recvbyte = recv(i, bufrecv,sizeof(bufrecv),0);
if(recvbyte <= 0)
{
FD_CLR(i,&readfds);
close(i);
continue;
}

printf("%s\n",bufrecv);

}
}
}
}

close(socket_fd);
close(accept_fd);
return 0;
}

轉(zhuǎn)載于:https://www.cnblogs.com/huiji12321/p/11380865.html

總結(jié)

以上是生活随笔為你收集整理的step5 . day6 网络编程 基于TCP协议的多并发模式(使用多进程、多线程、select函数分别实现)...的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。