循环服务器

一、服务器模型
  • 在网络程序里面,通常都是一个服务器处理多 个客户机。
  • 为了处理多个客户机的请求, 服务器端的程序有不同的处理方式。

1、循环服务器模型

socket();
bind();
liste();
while(1)
{
accept();
while(1)
{
recv
ret==0;
break;
}
close(acceptfd);
}
close(sockfd);

2、并发服务器模型

同一时刻相应多个客户端(tcp)。多进程模型/多线程模型/IO多路复用(select、poll、epoll)

socket();
bind();
listen()
while(1)
{
accept();
if(fork()==0)
{
    while(1)
    {
    recv
    ret==0;
    break;
    }
    close(acceptfd);
    exit();
    }   
     else
    {}}

注意:收到客户端消息后,打印下是来自哪个客户端的数据(来电显示)

使用SIGCHLD来处理子进程结束的信号,信号函数中回收进程资源。

3、多进程特点总结

  1. fork之前的代码被复制,但是不会重新执行一遍;fork之后的代码被复制,并且再被执行一遍。
  2. fork之后两个进程相互独立,子进程拷贝了父进程的所有代码,但内存空间独立
  3. fork之前打开文件,fork之后拿到的是同一个文件描述符,操作的是同一个文件指针

/*服务器创建代码 */
#include <stdio.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h> /* superset of previous */
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/wait.h>void handler(int arg)
{
    waitpid(-1, NULL, WNOHANG);
}
int main(int argc, char const *argv[])
{
    if (argc < 2)
    {
        printf("plase input <ip><port>\n");
        return -1;
    }
    //1.创建套接字,用于链接
    int sockfd;
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0)
    {
        perror("socket err");
        return -1;
    }
    printf("sockfd:%d\n", sockfd);
    //2.绑定 ip+port 填充结构体
    struct sockaddr_in saddr;
    saddr.sin_family = AF_INET;                   //协议族ipv4
    saddr.sin_port = htons(atoi(argv[1]));        //端口号,htons将无符号短整数hostshort从主机字节顺序到网络字节顺序。
    saddr.sin_addr.s_addr = inet_addr("0.0.0.0"); //ip地址,转化为16进制表示
    socklen_t len = sizeof(saddr);                //结构体大小
    //bind绑定ip和端口
    if (bind(sockfd, (struct sockaddr *)&saddr, len) < 0)
    {
        perror("bind err");
        return -1;
    }
    printf("bind success\n");
    //3.启动监听,把主动套接子变为被动套接字
    if (listen(sockfd, 6) < 0)
    {
        perror("listen err");
        return -1;
    }
    printf("listen success\n");
    //4.阻塞等待客户端的链接请求
    int acceptfd;
    char buf[64];
    int ret;
    while (1)
    {
        acceptfd = accept(sockfd, (struct sockaddr *)&saddr, &len);
        //获取客户端的ip和端口,(struct sockaddr *)&saddr:用来存放返回的ip,和端口
        if (acceptfd < 0)
        {
            perror("accept err");
            return -1;
        }
        printf("client ip:%s ,port:%d\n", inet_ntoa(saddr.sin_addr), ntohs(saddr.sin_port));
        printf("connect success\n");
        signal(SIGCHLD, handler);
        pid_t pid = fork(); //创建子进程
        if (pid < 0)
        {
            perror("fork err");
            return -1;
        }
        else if (pid == 0)
        {
            while (1)
            {
                ret = recv(acceptfd, buf, sizeof(buf), 0);
                if (strncmp(buf, "quit", 4) == 0) //接收到quit退出
                {
                    break;
                }
                if (ret < 0)
                {
                    perror("recv err.");
                    return -1;
                }
                else if (ret == 0) //客户端退出
                {
                    printf("client exit\n");                    break;
                }
                else
                {
                    printf("buf:%s\n", buf);
                }
            }
            close(acceptfd);//关闭子进程的文件描述副
            close(sockfd);//关闭子进程的套接字文件描述符,不影响主进程套接字
            exit(0);
        }
        close(acceptfd);//关闭主进程打开的文件描述符,
                        //为下次循环开辟的文件描述符空位置,否则只能连续开辟文件描述符到1024个
    }
    close(sockfd);
    return 0;
}

4、多线程模型

每来一个客户端连接,开一个子线程来专门处理客户端的数据,实现简单,占用资源较少,属于使用比较广泛的模型:

socket();
bind();
listen();
while(1)
{
accept();
pthread_creat();
pthread_detach();
}

1)多线程服务器

#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>void *mythread(void *arg)
{
    int acceptfd= *((int *)arg);
    int ret;
    char buf[128];
    while (1)
    {
        ret=recv(acceptfd,buf,sizeof(buf),0);
        if (ret < 0)
        {
            perror("recv err.");
            return -1;
        }else if (ret ==0)
        {
            printf("%d client exit\n",acceptfd);
            close(acceptfd);
            break;
        }else
        {
            printf("buf:%s\n",buf);
        }
    }
    pthread_exit(NULL);
}int main(int argc, char const *argv[])
{    if (argc != 2)
    {
        printf("please input %s <port>\n", argv[0]);
        return -1;
    }
    // 1.创建流式套接字
    int sockfd = socket(AF_INET, SOCK_STREAM, 0); // 链接
    if (sockfd < 0)
    {
        perror("socket err.");
        return -1;
    }
    printf("sockfd:%d\n", sockfd); // 3
    // 填充ipv4的通信结构体    struct sockaddr_in saddr, caddr;
    saddr.sin_family = AF_INET;
    saddr.sin_port = htons(atoi(argv[1])); //"8888" int a= atoi("8888")//a=8888
    // saddr.sin_addr.s_addr = inet_addr(argv[1]);    // 设置服务器自动获取自己主机的ip
    //  saddr.sin_addr.s_addr = htonl(INADDR_ANY);//INADDR_ANY  0x00000000 "0.0.0.0"
    //  saddr.sin_addr.s_addr = INADDR_ANY;
    saddr.sin_addr.s_addr = inet_addr("0.0.0.0");    socklen_t len = sizeof(caddr);    // 2.绑定套接字 ip和端口(自己)
    if (bind(sockfd, (struct sockaddr *)&saddr, sizeof(saddr)) < 0)
    {
        perror("bind err.");
        return -1;
    }
    printf("bind ok.\n");    // 3.监听
    if (listen(sockfd, 5) < 0)
    {
        perror("listen err.");
        return -1;
    }
    printf("listen ok.\n");
    while (1)
    {
        int acceptfd = accept(sockfd, (struct sockaddr *)&caddr, &len);
        if (acceptfd < 0) // 4
        {
            perror("accept err.");
            return -1;
        }
        printf("acceptfd=%d\n", acceptfd);
        printf("ip=%s port=%d\n", inet_ntoa(caddr.sin_addr), ntohs(caddr.sin_port));
        pthread_t tid;
        pthread_create(&tid,NULL,mythread,&acceptfd);
        pthread_detach(tid);//游离态
    }
    close(sockfd);
    return 0;
}

5、IO多路复用模型

借助select、poll、epoll机制,将新连接的客户端描述符增加到描述符表中,只需要一个线程即可处理所有的客户端连接,在嵌入式开发中应用广泛,不过代码写起了稍显繁琐。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/174771.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【C++11/高级语法】bind绑定器和function函数对象

目录 bind1st和bind2nd什么时候会用到bind1st和bind2nd的底层实现原理function函数对象类型的应用示例lambda表达式的应用实践 橙色 绑定器和函数对象operator() 函数对象就是对象拥有()运算符重载函数&#xff0c;这个对象使用起来就跟函数调用特别相似。 1.C STL中的绑定器…

前端学习笔记--面试题系列总结

event loop它的执行顺序&#xff1a; 一开始整个脚本作为一个宏任务执行执行过程中同步代码直接执行&#xff0c;宏任务进入宏任务队列&#xff0c;微任务进入微任务队列当前宏任务执行完出队&#xff0c;检查微任务列表&#xff0c;有则依次执行&#xff0c;直到全部执行完执…

Hadoop入门——数据分析基本步骤

文章目录 1.概述2.分析步骤2.1第一步 明确分析目的和思路2.2第二步 数据收集2.3第三步 数据处理2.4第四步 数据分析2.5第五步 数据展现2.6第六步 报告撰写 3.总结 1.概述 2.分析步骤 2.1第一步 明确分析目的和思路 2.2第二步 数据收集 2.3第三步 数据处理 2.4第四步 数据分析 …

如何使用Docker搭建Drupal内容管理系统并远程访问

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525;个人专栏:《Linux深造日志》《C干货基地》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 前言1. Docker安装Drupal2. 本地局域网访问3 . Linux 安装cpolar4. 配置Drupal公网访问地址5. 公网远程访问Drupal…

Kakaotalk如何注册?常见问题解答

kakaoTalk是一款韩国即时通讯软件&#xff0c;使用程度类似于国内的微信&#xff0c;他还包含叫车服务、食品外送、餐厅预订、支付和游戏等多种功能&#xff0c;几乎每个韩国人都在使用KakaoTalk。 因此&#xff0c;对于要发展韩国市场的独立站人&#xff0c;这款软件必定是营…

人大金仓物理备份异机恢复

概述 KingbaseES V8支持使用RMAN物理备份在异机环境恢复&#xff0c;通过重新克隆方式完扩展主备集群。 原集群环境&#xff1a;演示用例&#xff0c;仅供参考 查看原集群备份和物理备份路径 异机恢复 前置条件 *获取原集群物理备份文件&#xff0c;包括全量备份、增量备份…

Java项目maven打包,打jar包中不包含项目引用第三方jar包,以及打war包不能将其放到lib的问题

在使用maven进行打包项目中&#xff0c;想要将第三方的jar包放入&#xff0c;有两种方法&#xff1a;一种将jar包上传到maven库中&#xff0c;第二种再pom.xml中进行配置&#xff0c;第三种 情况是需要打包成war包放入tomcat中&#xff1b;具体如下&#xff1a; 第一种&#x…

nacos集群配置(超完整)

win配置与linux一样&#xff0c;换端口或者换ip&#xff0c;文章采用的 linux不同IP&#xff0c;同一端口 节点ipportnacos1192.168.253.168848nacos2192.168.253.178848nacos3192.168.253.188848 单IP多个端口 1.复制两个&#xff0c;重命名 2.修改 conf目录下的 application…

Ubuntu 20.04编译Chrome浏览器

本文记录chrome浏览器编译过程&#xff0c;帮助大家避坑qaq 官网文档&#xff1a;https://chromium.googlesource.com/chromium/src//main/docs/linux/build_instructions.md 一.系统要求 一台64位的英特尔机器&#xff0c;至少需要8GB的RAM。强烈推荐超过16GB。至少需要100…

淘宝京东优惠券信息API接口系列

获取淘宝优惠券信息接口需要使用淘宝开放平台提供的API接口。以下是获取优惠券信息的步骤&#xff1a; 进入淘宝开放平台&#xff0c;注册并登录账号。在开放平台页面中&#xff0c;找到“优惠券”或“营销工具”等相关的API接口&#xff0c;根据需要进行选择。根据接口文档&a…

Java 设计模式——组合模式

目录 1.概述2.结构3.实现3.1.抽象构件3.2.容器构件3.3.叶子节点3.4.测试 4.分类5.使用场景6.优点 1.概述 &#xff08;1&#xff09;大家对于上面这个图片肯定非常熟悉&#xff0c;上图我们可以看做是一个文件系统&#xff0c;对于这样的结构我们称之为树形结构。在树形结构中可…

通讯协议学习之路(实践部分):SPI开发实践

通讯协议之路主要分为两部分&#xff0c;第一部分从理论上面讲解各类协议的通讯原理以及通讯格式&#xff0c;第二部分从具体运用上讲解各类通讯协议的具体应用方法。 后续文章会同时发表在个人博客(jason1016.club)、CSDN&#xff1b;视频会发布在bilibili(UID:399951374) 本文…