实现TCP服务器端多路并发的方法有①多进程②多线程③IO多路复用(select poll epoll函数)

1.多进程实现并发模式(仅在服务器端更改之前代码实现)

服务器端
#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.创建套接字文件,用于链接
int socket_fd;
socket_fd = socket(AF_INET,SOCK_STREAM,0);
if(socket_fd == -1){
perror("socket");
return -1;
}
printf("socket Created ok\n");

//填充结构体sockaddr_in,用于传参
int bind_fd;
struct sockaddr_in serveraddr,connectaddr; //定义填充的结构体
bzero((void *)&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET; //IPV4协议
serveraddr.sin_port = htons(atoi(argv[1])); //本地端口号转化位网络字节码
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);//本地IP转化位网络字节码,任意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.监听,将套接字文件属性该主动为被动模式
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.多线程实现并发

#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.创建套接字文件,用于链接
int socket_fd;
socket_fd = socket(AF_INET,SOCK_STREAM,0);
if(socket_fd == -1){
perror("socket");
return -1;
}
printf("socket Created ok\n");

//填充结构体sockaddr_in,用于传参
int bind_fd;
struct sockaddr_in serveraddr,connectaddr; //定义填充的结构体
bzero((void *)&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET; //IPV4协议
serveraddr.sin_port = htons(atoi(argv[1])); //本地端口号转化位网络字节码
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);//本地IP转化位网络字节码,任意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.监听,将套接字文件属性该主动为被动模式
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函数实现多并发

#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.创建套接字文件,用于链接
int socket_fd;
socket_fd = socket(AF_INET,SOCK_STREAM,0);
if(socket_fd == -1){
perror("socket");
return -1;
}
printf("socket Created ok\n");

//填充结构体sockaddr_in,用于传参
int bind_fd;
struct sockaddr_in serveraddr,connectaddr; //定义填充的结构体
bzero((void *)&serveraddr, sizeof(serveraddr));
serveraddr.sin_family = AF_INET; //IPV4协议
serveraddr.sin_port = htons(atoi(argv[1])); //本地端口号转化位网络字节码
serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);//本地IP转化位网络字节码,任意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.监听,将套接字文件属性该主动为被动模式
int listen_fd;
listen_fd = listen(socket_fd,10);

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

//创建监听集合
fd_set readfds,tempfds;
//清空集合
FD_ZERO(&readfds);
FD_ZERO(&tempfds);
//将关心的文件描述符加入集合
FD_SET(0,&readfds);
FD_SET(socket_fd,&readfds);

//设置最大监听个数
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.阻塞函数,等待客户链接请求,不关心链接的是那个客户端,填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;
}

转载于:https://www.cnblogs.com/huiji12321/p/11380865.html

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

  1. 网络编程——基于TCP协议的通讯录【课程设计】

    网络编程--基于TCP协议的通讯录[课程设计] 本文目录 网络编程--基于TCP协议的通讯录[课程设计] 一.设计题目和要求 设计目标: 课程设计系统组成及模块功能: 二.设计内容 服务端 客户端 S ...

  2. step5 . day2 网络编程 基于TPC协议的网络编程流程及API

    基于TCP协议的网络编程流程 服务器端: 1) socket(),创建套接字文件,用于连接 2) bind(), 绑定,把socket()函数返回的文件描述符和IP.端口号进行绑定; 3) liste ...

  3. step5 . day4 网络编程 基于UDP协议的网络编程流程及API

    UDP协议是面向无连接,不可靠的传输协议,编程流程如下; 服务器端: 1. socket(),返回一个文件描述符,用于通信 2. bind(); //得先完成结构体的填充sockaddr_in 3. ...

  4. step5 . day3 网络编程 基于TPC协议的网络编程Demo,类FTP功能

    1.客户端 //cilent code #include <stdio.h> #include <sys/types.h> #include <sys/socket.h& ...

  5. step5 . day5 网络编程 基于UDP协议的多人网络在线聊天功能

    模拟在线群聊功能,使用多进程完成聊天内容的接受和服务器端的转发,demo代码记录参考 //client_chat_UDP code #include <stdio.h> #include ...

  6. 基于ZYNQ-7000的AI加速器设计之PS端(ARM)网络编程(TCP协议)

    前注:ARM端的TCP协议编程步骤和UDP协议编程步骤完全相同,只是在ARM端的C代码实现不同,在本次TCP协议实现过程中我们主要利用了官方提供的Demo,然后根据自己的需要加以改写,具体过程如下. ...

  7. Linux网络编程——基于tcp/ip的模拟聊天(文件传输)工具

    开发平台:Linux 开发工具:Ubuntu, sourceInsight4.0 项目介绍: 本项目基于TCP/IP协议创建一个网络通信系统,可以实现客户之间的聊天通信以及文件传输,同时利用进程实现多 ...

  8. Java网络编程——基于UDP协议的聊天室

    UDP简述    UDP(User Datagram Protocol)协议是Internet 协议集支持的一个无连接的传输协议,中文名为用户数据报协议.它为应用程序提供了一种无需建立连接就可以发送封 ...

  9. Linux 系统应用编程——网络编程(TCP 协议三次握手过程)

    TCP(Transmission Control Protocol) 传输控制协议  TCP是主机对主机层的传输控制协议,提供可靠的连接服务,采用三次握手确认建立一个连接: 位码即tcp标志位,有6种 ...

最新文章

  1. 入门视频采集与处理(学会分析YUV数据)
  2. 最小生成树板子-AcWing 858. Prim算法求最小生成树
  3. 夸克浏览器怎么安装脚本_还你清爽流畅!这五款手机浏览器!黑马强推
  4. 记一次SAP开发工程师给微软Azure报incident的体验
  5. XXE漏洞利用技巧(XML注入):从XML到远程代码执行
  6. 聚类方法:DBSCAN算法研究
  7. NLP(paper + code)
  8. java.lang.NoClassDefFoundError: javax/mail/internet/AddressException解决
  9. 空降了一位CTO,不到3个月,就被发现是一地鸡毛
  10. 亚马逊推出人工智能咨询服务,机器学习B2B业务又迈出一步
  11. GNU make manual 翻译( 一百八十二)
  12. 学python能做什么-非计算机专业的人学python能做什么?
  13. (开源)微信小程序实时控制stc89c51,通过esp8266
  14. android面板驱动的使用方法,Android 专用驱动之Ashmen
  15. Linux进程管理和服务控制
  16. 使用VMware虚拟机通过Panabit抓取爱快虚拟机PPPoE报文
  17. 小树corexy改voron装前必看
  18. 2019年蚂蚁金服面经(已拿Offer)!附答案!!
  19. ENG Note 1 : 英语写作模板
  20. 【在表格下方创建“添加”按钮,完成添加用户信息,点击后自动添加等功能】

热门文章

  1. 计算机网络接入接口设备,一种计算机网络集成接口设备的制作方法
  2. r语言员工离职_HR,你真的会做员工面谈吗?
  3. 最新详细的JMM内存模型(三天熬夜血肝)
  4. 此域的推荐安全级别是“安全级-高”的解决办法
  5. C++知识 interview
  6. bit索引 mysql_Mysql优化之索引实现原理
  7. c++ 12.一维数组冒泡排序
  8. [BUUCTF-pwn]——ciscn_2019_n_8
  9. php裁剪图片白边,php缩略图填充白边的示例代码
  10. 广西2021高考成绩什么时候可以查询,2021年广西高考成绩什么时候公布出来,几月几号几点钟可以查询...