并发服务器

news2025/1/8 5:44:52

服务器,后台服务软件,后端开发,为软件客户端提供支持,数据支持,存储支持,数据中转

高并发要求,随着用户数量的增长,服务器的设计结构扩展,为更多用户提供服务器。

Web服务器(网页网站的后端都是web服务器),软件服务器(手机软件或者pc软件)。

开源服务器软件Apache,Nginx,用户配置即可使用,提供web服务器。

关于网页网站二点技术标准:共用协议(http https),超文本标记语言html,端口号共享使用http=80 https=443。

软件服务器,因不同软件不同功能,服务器的结构也不同,所以很难开源,因为并不适用其他软件。

后端开发的操作系统(服务器操作系统),Linux,Unix,window server,大多数都是控制台系统。

服务器集群:1.处理服务器(128核),2.图形处理器(GPU)渲染或图形计算,3.数据库服务器 多版本数据库兼容。4.文件存储系统。

搭建集群的成本比较高。

服务器的核心任务,网络穿透。

服务器作为信息中转站,帮助所有用户转接数据,变更网络环境,用户重新连接服务器,重新提交信息,服务器实时更新客户端的网络信息。

代理服务器

请求与响应(Request/Response)一次交互。

代理服务器可以作为过滤层,部署防火墙,防DOS攻击等等,隐藏主服务器。

横向扩展服务器硬件,分布式结构。分布式管理程序:资源共享,资源利用最大化,让每台处理机统一分配管理与调度,成为一个个体。

负载均衡概念:将大量的业务请求均匀分发给n个处理单元。避免某些处理单元过载,避免某些处理单元长期闲置。

轮询分发(逐个分发)

根据处理单元的负载情况,决定如何分配。

心跳机制 Keep_alive

心跳测试,在一个时间段内,要通过心跳测试测试对端存活是否有效,有效保持连接,无效断开连接。

操作系统自带心跳机制(setsocketopt),这个心跳机制默认下要很久才能检测出对端是否断开。

如果一个连接,一段时间没有交互,会被冷处理,通过间隔收发心跳包,让连接保持活性,避免被冷处理。

单进程服务器

连接部分,业务处理部分,协议解析部分。连接部分和业务处理部分都是阻塞的,请求时无法连接,等待连接时无法读取请求。

单进程服务端,满足基本tcp连接,完成简单的数据交互,对于系统时间的请求和响应。

htons();将 16 位的短整数(如端口号)从主机字节顺序转换为网络字节顺序。

htonl();将 32 位整数从主机字节顺序转换为网络字节顺序。

INADDR_ANY,服务器将绑定到所有可用的网络接口上。这意味着服务器将接受任何网卡或接口上的连接,而不限定于一个特定的 IP 地址。

bind()用于将一个套接字绑定到一个特定的地址和端口。绑定操作告诉操作系统,当有数据到达这个特定的 IP 地址和端口时,应该将数据交给这个套接字处理。

单进程阻塞

服务端:

#include<stdio.h>
#include<sys/fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<time.h>
#include<string.h>
#include<arpa/inet.h>
#include<sys/socket.h>
#include<unistd.h>
#include<stdlib.h>

int init_net()
{
    int sock;
    struct sockaddr_in sockaddr;
    bzero(&sockaddr,sizeof(sockaddr));
    sockaddr.sin_family=AF_INET;
    sockaddr.sin_port=htons(1234);
    sockaddr.sin_addr.s_addr=htonl(INADDR_ANY);
    if((sock=socket(AF_INET,SOCK_STREAM,0))==-1)
        printf("socket error\n");
    else printf("socket ok\n");
    if((bind(sock,(struct sockaddr*)&sockaddr,sizeof(sockaddr)))==-1)
    {
        perror("bind error\n");
    }
    else printf("bind ok\n");
    listen(sock,128);
    return sock;
}

void mess_response(int so,struct sockaddr_in addr)
{
    char s[1024];
    char c[16];
    bzero(c,16);
    bzero(s,1024);
    inet_ntop(AF_INET,&addr.sin_addr,c,16);
    sprintf(s,"hi %s\n",c);
    int t=send(so,s,strlen(s),MSG_NOSIGNAL);
    if(t<0)
    {
    perror("err");
    }
    return;
}

int recv_request(int sock)
{
    char s[1024];
    char tim[1024];
    bzero(s,sizeof(s));
    bzero(tim,sizeof(tim));
    int len;
    time_t tp;
    while((len=recv(sock,s,sizeof(s),0))>0)
    {
    s[strlen(s)-1]='\0';
    if(strncmp(s,"time",strlen(s))==0)
    {
        tp=time(NULL);
        ctime_r(&tp,tim);
        send(sock,tim,strlen(tim),MSG_NOSIGNAL);
    }
    else{
        send(sock,"try again",9,MSG_NOSIGNAL);
    }
    bzero(s,sizeof(s));
    bzero(tim,sizeof(tim));
    }
    if(len==0)
    {
        printf("client sock %d its exiting...\n",sock);
        close(sock);
 }
}
int main()
{
    int sock,csock;
    struct sockaddr_in sock_client;
    bzero(&sock_client,sizeof(sock_client));
    sock=init_net();
    printf("init ok\n");
    socklen_t socklen;
    while(1)
    {
    socklen=sizeof(sock_client);
   if(( csock=(accept(sock,(struct sockaddr*)&sock_client,&socklen)))==-1)
    perror("err\n");
   else{
       printf("accept success\n");
   }
    mess_response(csock,sock_client);
    recv_request(csock);
    }
    close(csock);
    close(sock);
    return 0;
}

客户端:

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<arpa/inet.h>
#include<signal.h>
#include<sys/socket.h>
#include<sys/fcntl.h>
#include<string.h>
int main()
{
    int sock;
    struct sockaddr_in serveraddr;
    serveraddr.sin_family=AF_INET;
    serveraddr.sin_port=htons(1234);
    inet_pton(AF_INET,"127.0.0.1",&serveraddr.sin_addr.s_addr);
    if((sock=socket(AF_INET,SOCK_STREAM,0))==-1)
    {
        printf("socket error\n");
    }
    else printf("socket success\n");
    if((connect(sock,(struct sockaddr*)&serveraddr,sizeof(serveraddr)))==-1)
    {
        printf("connect error\n");
    }
    else{
        printf("connect success\n");
    }
    char buff[1024];
    bzero(buff,1024);
    int r=recv(sock,buff,sizeof(buff),0);
    printf("%d\n",r);
    //if((recv(sock,buff,sizeof(buff),0))==-1)
    //{
      //  perror("recv error\n");
    //}
    //else printf("recv ok\n");
    printf("%s\n",buff);
 bzero(buff,1024);
    while((fgets(buff,sizeof(buff),stdin))!=NULL)
    {
        send(sock,buff,strlen(buff),MSG_NOSIGNAL);
        bzero(buff,1024);
        recv(sock,buff,sizeof(buff),0);
        printf("%s\n",buff);
    }
    close(sock);
    return 0;
}

单进程非阻塞轮询模型

服务端:

#include<stdio.h>
#include<sys/fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<time.h>
#include<string.h>
#include<arpa/inet.h>
#include<sys/socket.h>
#include<unistd.h>
#include<stdlib.h>
#include<fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<errno.h>
int clientsock[10000];
int init_net();
void mess_response(int,struct sockaddr_in);
int recv_request(int);
int start_server();


#include<server.h>
void mess_response(int so,struct sockaddr_in addr)
{
        char s[1024];
        char c[16];
        bzero(c,16);
        bzero(s,1024);
        inet_ntop(AF_INET,&addr.sin_addr,c,16);
         sprintf(s,"hi %s\n",c);
          int t=send(so,s,strlen(s),MSG_NOSIGNAL);
          if(t<0)perror("err");
          return;
}


#include<server.h>
int recv_request(int sock)
{
        char s[1024];
        char tim[1024];
        bzero(s,sizeof(s));
        bzero(tim,sizeof(tim));
        int len;
        time_t tp;
        int i=0;
        for(i;i<10000;i++)
        {
        if(clientsock[i]!=-1)
        {
        if((len=recv(clientsock[i],s,sizeof(s),MSG_DONTWAIT))>0)
        {
        s[strlen(s)-1]='\0';
        if(strncmp(s,"time",strlen(s))==0)
        {
                tp=time(NULL);
                ctime_r(&tp,tim);
                send(clientsock[i],tim,strlen(tim),MSG_NOSIGNAL);
        }
        else{
                send(clientsock[i],"try again",9,MSG_NOSIGNAL);
        }
        bzero(s,sizeof(s));
        bzero(tim,sizeof(tim));
        }
        if(len==0)
        {
                printf("client sock %d its exiting...\n",sock);
                close(clientsock[i]);
                clientsock[i]=-1;
        }
        }
        }
}


#include<server.h>
int server_start()
{
        int sock,csock;
        struct sockaddr_in sock_client;
        int i;
        for(i=0;i<10000;i++)
        clientsock[i]=-1;
        bzero(&sock_client,sizeof(sock_client));
        sock=init_net();
        printf("init ok\n");
        int flag;
        fcntl(sock,F_GETFL,&flag);
        flag|=O_NONBLOCK;
        fcntl(sock,F_SETFL,flag);
        socklen_t socklen;
        while(1)
        {
        socklen=sizeof(sock_client);
        if(( csock=(accept(sock,(struct sockaddr*)&sock_client,&socklen)))>0)
           {
                for(i=0;i<10000;i++)
           {
                if(clientsock[i]==-1)
                {
                clientsock[i]=csock;
                break;
                }
           }
           mess_response(csock,sock_client);
           printf("accept success\n");
         }
        if(csock==-1)
        {
        if(errno==EAGAIN)
        recv_request(csock);
        else perror("accept call failed\n");
        }
        }
        close(csock);
        close(sock);
}


#include<server.h>
int init_net()
{
int sock;
struct sockaddr_in sockaddr;
bzero(&sockaddr,sizeof(sockaddr));
sockaddr.sin_family=AF_INET;
sockaddr.sin_port=htons(1234);
sockaddr.sin_addr.s_addr=htonl(INADDR_ANY);
if((sock=socket(AF_INET,SOCK_STREAM,0))==-1)
 printf("socket error\n");
 else printf("socket ok\n");
if((bind(sock,(struct sockaddr*)&sockaddr,sizeof(sockaddr)))==-1)
            {perror("bind error\n");}
 else printf("bind ok\n");
     listen(sock,128);
             return sock;
}


#include<server.h>
int main()
{
server_start();
return 0;
}

单进程服务端只符合简易环境下的需求。

单进程采用非阻塞交替执行的策略让等待连接和读取处理交替执行,但是如果服务器长时间为某个客户端处理业务,导致无法建立新连接处理其他人的数据。

并发服务器-多进程服务器

多进程服务器是具备并发处理能力的服务器模型,可以并发连接并发处理,为若干客户端提供服务。并发数量取决于进程数量,频繁创建和销毁进程有庞大的系统开销。

多进程模型中有多个处理单元,为多对多模型。多进程模型中,子进程随客户端持续,链接成功创建,客户端退出则子进程退出。僵尸进程回收,无论是阻塞回收还是非阻塞,都会影响连接,所以我们让线程负责回收。

#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<time.h>
#include<string.h>
#include<arpa/inet.h>
#include<sys/socket.h>
#include<fcntl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<errno.h>
#include<pthread.h>
#include<signal.h>
#include<sys/wait.h>

int init_net();
void mess_response(int,struct sockaddr_in);
int recv_request(int);
int start_server();
void* thread_wait(void* arg);
void sig_wait(int n);


#include<server.h>
int init_net()
{
int sock;
struct sockaddr_in sockaddr;
bzero(&sockaddr,sizeof(sockaddr));
sockaddr.sin_family=AF_INET;
sockaddr.sin_port=htons(1234);
sockaddr.sin_addr.s_addr=htonl(INADDR_ANY);
if((sock=socket(AF_INET,SOCK_STREAM,0))==-1)
 printf("socket error\n");
 else printf("socket ok\n");
if((bind(sock,(struct sockaddr*)&sockaddr,sizeof(sockaddr)))==-1)
            {perror("bind error\n");}
 else printf("bind ok\n");
     listen(sock,128);
             return sock;
}

#include<server.h>
void mess_response(int so,struct sockaddr_in addr)
{
        char s[1024];
        char c[16];
        bzero(c,16);
        bzero(s,1024);
        inet_ntop(AF_INET,&addr.sin_addr,c,16);
         sprintf(s,"hi %s\n",c);
          int t=send(so,s,strlen(s),MSG_NOSIGNAL);
          if(t<0)perror("err");
          return;
}

#include<server.h>
int recv_request(int sock)
{
        char s[1024];
        char tim[1024];
        bzero(s,sizeof(s));
        bzero(tim,sizeof(tim));
        int len;
        time_t tp;

        while((len=recv(sock,s,sizeof(s),0))>0)
        {
        s[strlen(s)-1]='\0';
        if(strncmp(s,"time",strlen(s))==0)
        {
                tp=time(NULL);
                ctime_r(&tp,tim);
                send(sock,tim,strlen(tim),MSG_NOSIGNAL);
        }
        else{
                send(sock,"try again",9,MSG_NOSIGNAL);
        }
        bzero(s,sizeof(s));
        bzero(tim,sizeof(tim));
        }
                printf("client sock %d  ,child process %d its exiting...\n",sock,getpid());
                close(sock);
                exit(sock);
return 0;
}

#include<server.h>
void sig_wait(int n)
{
pid_t zpid;
while((zpid=waitpid(-1,NULL,WNOHANG))!=-1)
{
if(zpid>0){
printf("thread wait success,zpid %d\n",zpid);
}
}}


#include<server.h>
void* thread_wait(void* arg)
{
struct sigaction act,oact;
pthread_detach(pthread_self());
act.sa_handler=sig_wait;
act.sa_flags=0;
sigemptyset(&act.sa_mask);
sigaction(SIGCHLD,&act,&oact);
sigprocmask(SIG_SETMASK,&act.sa_mask,NULL);
printf("wait thread 0x%x waiting..\n",(unsigned int)pthread_self());
while(1)
sleep(1);
pthread_exit(NULL);
}


#include<server.h>
int server_start()
{
        int sock,csock;
        struct sockaddr_in sock_client;
        int i;
        bzero(&sock_client,sizeof(sock_client));
        sock=init_net();
        printf("init ok\n");
        socklen_t socklen;

        sigset_t set,oset;
        sigemptyset(&set);
        sigaddset(&set,SIGCHLD);
        sigprocmask(SIG_SETMASK,&set,&oset);
        pthread_t tid;
        int err;
        if((err=pthread_create(&tid,NULL,thread_wait,NULL))>0){
        printf("wait thread create err:%s\n",strerror(err));
        exit(0);
        }
        pid_t pid;
        while(1)
        {
        socklen=sizeof(sock_client);
        if(( csock=(accept(sock,(struct sockaddr*)&sock_client,&socklen)))>0)
        {
            pid=fork();
            if(pid>0)
           {
          mess_response(csock,sock_client);
           printf("accept success\n");
         }
           else if(pid==0)
        {
        recv_request(csock);
        }
        else exit(0);
        }
        else {
        printf("accept error\n");
                }
       }
        close(sock);
        return 0;
}


#include<server.h>
int main()
{
server_start();
return 0;
}

多进程稳定性强,因为每个处理单元是一个进程,一个进程异常退出不会影响其他进程。

并发服务器-多线程服务器

稳定性较差,线程崩溃可能影响其他线程,但是开销小轻量级。(线程安全问题)

并发量取决于线程数量。

频繁创建销毁进程有庞大的系统开销。

和多进程差不多。

多路IO复用服务器

IO复用技术又名多路IO转接技术,可以帮助开发者监听大量的sock。

sock监听技术,io监听技术。

select模型

写的是单进程的。

#include<sys/select.h>//头文件

fd_set set;//监听集合类型

FD_ZERO(&set);//初始化监听集合,将位码初始化为0

FD_SET(int sockfd,&set);//将sockfd在集合中对应的位设置成1

FD_CLR(int sockfd,&sest);//将sockfd在集合中的对应位设置为0

int code=FD_ISSET(int sock,&set);//返回sockfd在集合中位码

int ready=select(maxfd+1,fd_set* rd,fd_set* wr,fd_set* err,timeval* timeout);//监听函数,监听事件选择以集合为单位,批处理,timeout=NULL,阻塞监听,监听集合对应文件描述符表,所以在设置监听数量时要考虑设备描述符。

struct timeval* val;   val.secons=0; val.ms=0;//如果要非阻塞使用select,需要定义时间结构体并将时间成员初始化成0,如果需要select定时阻塞,用户自行设置时间在结构体中。

#include<unistd.h>
#include<stdlib.h>
#include<signal.h>
#include<time.h>
#include<sys/select.h>
#include<sys/socket.h>
#include<string.h>
#include<arpa/inet.h>
int main()
{
        //sock的就绪数量
        int socknum;
        //存储客户端sock
        int sockarry[1020];
        //最大的描述符
        int maxfd;
        int ssock;
        int csock;
        //初始化客户端sock数组
        int i;
        for(i=0;i<1020;i++)
        {
        sockarry[i]=-1;
        }
        //初始化网络
        struct sockaddr_in saddr,caddr;
        saddr.sin_family=AF_INET;
        saddr.sin_port=htons(1234);
        saddr.sin_addr.s_addr=htonl(INADDR_ANY);
        if((ssock=socket(AF_INET,SOCK_STREAM,0))==-1)
        {
        perror("sock init error\n");
        }
        bind(ssock,(struct sockaddr*)&saddr,sizeof(saddr));
        listen(ssock,128);
        maxfd=ssock;
        //监听集合
        fd_set set,oset;
        //初始化监听集合
        FD_ZERO(&set);
        FD_SET(ssock,&set);
        socklen_t addrlen;
        //初次连接回复
        char response[1500];
        char cip[16];
        char buff[1024];
        bzero(response,1500);
        bzero(cip,16);
        bzero(buff,1024);
        //时间
        char timebuff[1024];
        time_t tp;

        int len;
        printf("server running\n");
        //轮询监听
        while(1)
        {
        oset=set;
        if((socknum=select(maxfd+1,&oset,NULL,NULL,NULL))==-1)
        perror("select() failed");
    while(socknum)
        {
        //是severfd
        if(FD_ISSET(ssock,&oset))
        {
        addrlen=sizeof(caddr);
        csock=accept(ssock,(struct sockaddr*)&caddr,&addrlen);
        //首次连接成功返回信息
        inet_ntop(AF_INET,&caddr.sin_addr.s_addr,cip,16);
        sprintf(response,"hi %s",cip);
        send(csock,response,strlen(response),MSG_NOSIGNAL);
        bzero(response,sizeof(response));
        //设置监听,存入数组
        FD_SET(csock,&set);
        for( i=0;i<1020;i++)
        {
        if(sockarry[i]==-1)
        {
        sockarry[i]=csock;
        break;
        }
        }
        //更新最大sockfd
        if(maxfd<csock)maxfd=csock;
        //将oset设置为0
        FD_CLR(ssock,&oset);
        }
        //是clientfd
        else
        {
        i=0;
        //遍历sockarry数组,查找就绪的csock并处理
        for(i;i<1020;i++)
        {
        if(sockarry[i]!=-1)
        if(FD_ISSET(sockarry[i],&oset))
        {
        //读取处理数组
        if((len=recv(sockarry[i],buff,sizeof(buff),0))>0)
        {
        buff[strlen(buff)-1]='\0';
        if((strcmp(buff,"time"))==0)
        {
        bzero(timebuff,1024);
        tp=time(NULL);
        ctime_r(&tp,timebuff);
        send(sockarry[i],timebuff,strlen(timebuff),MSG_NOSIGNAL);
        }
        else{
        send(sockarry[i],"try again",10,MSG_NOSIGNAL);
        }
        bzero(buff,1024);
        FD_CLR(sockarry[i],&oset);
        }
        if(len==0)
        {
        //客户端退出,在监听中删除,在组中删除
        FD_CLR(sockarry[i],&set);
        FD_CLR(sockarry[i],&oset);
        close(sockarry[i]);
        sockarry[i]=-1;
        }
        break;
        }
        }
        }
        socknum--;
        }
        }
        close(ssock);
        return 0;
}

优点:

1.使用比较简单,了解监听集合以及IO复用机制即可使用,帮助用户完成少量sock的网络事件监听

2.跨平台兼容性比较好,在各个系统语言均有select支持

3.select支持微妙级定时,可以满足一些特定需求

缺点:

1.select无法满足大监听需求,最大监听数1024

2.轮询监听,随着轮询数量的增大,io处理性能呈线性下降

3.select监听到就绪后只返回就绪的数量,需要用户自行遍历查找就绪的sock

4.需要用户进行传入传出分离设置

5.随着select的持续使用,会有庞大的开销和挂载开销

6.监听数目比较少,其次设置监听不灵活,无法针对不同的sock设置不同的监听

poll模型

优点:

1.监听事件的种类丰富,对监听与就绪进行了传入传出分离,无需用户分离

2.poll支持用户自定义长度结构体数组作为集合,突破了1024限制(轮询)

缺点:

1.轮询问题

2.拷贝开销挂载开销

3.只返回就绪需要用户自行遍历查找就绪的sock

4.只支持毫秒级别定时

5.在某些特定的Linux版本才可以使用

struct pollfd listen_array[4096];//监听数组,还可以帮助用户存储sock

listen_array[0].fd=sock;//如果需要监听则设置为某个sock,取消监听设置为-1

listen_arry[0].events=POLLIN|POLLOUT|POLLERR;//设置要监听的事件

listen_arry[0],revents;//如果监听的sock就绪,系统将就绪事件传到revents中

int readycode=poll(listen_array);//timeout -1阻塞 0非阻塞 >0定时阻塞

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<signal.h>
#include<time.h>
#include<sys/select.h>
#include<sys/socket.h>
#include<poll.h>
#include<string.h>
#include<arpa/inet.h>
int main()
{
        //sock的就绪数量
        int socknum;
        //存储客户端sock
        struct pollfd sockarry[4096];
        int ssock;
        int csock;
        //初始化客户端sock数组
        int i;
        for(i=0;i<4096;i++)
        {
        sockarry[i].fd=-1;
        sockarry[i].events=POLLIN;
        }
        //初始化网络
        struct sockaddr_in saddr,caddr;
        saddr.sin_family=AF_INET;
        saddr.sin_port=htons(1234);
        saddr.sin_addr.s_addr=htonl(INADDR_ANY);
        if((ssock=socket(AF_INET,SOCK_STREAM,0))==-1)
        {
        perror("sock init error\n");
        }
        bind(ssock,(struct sockaddr*)&saddr,sizeof(saddr));
        listen(ssock,128);
         //监听集合
        sockarry[0].fd=ssock;
        socklen_t addrlen;
        //初次连接回复
        char response[1500];
        char cip[16];
        char buff[1024];
        bzero(response,1500);
        bzero(cip,16);
        bzero(buff,1024);
        //时间
        char timebuff[1024];
        time_t tp;

        int len;
        printf("server running\n");
        //轮询监听
        while(1)
        {
        if((socknum=poll(sockarry,4096,-1))==-1)
        perror("poll() failed");
        while(socknum)
        {
        //是severfd
        if(sockarry[0].revents==POLLIN)
        {
        addrlen=sizeof(caddr);
        csock=accept(ssock,(struct sockaddr*)&caddr,&addrlen);
        //首次连接成功返回信息
        inet_ntop(AF_INET,&caddr.sin_addr.s_addr,cip,16);
        sprintf(response,"hi %s",cip);
        send(csock,response,strlen(response),MSG_NOSIGNAL);
        bzero(response,sizeof(response));
        //设置监听,存入数组
        for( i=1;i<4096;i++)
        {
        if(sockarry[i].fd==-1)
        {
        sockarry[i].fd=csock;
        break;
        }
        }
        //就绪处理
        sockarry[0].revents=0;
        }
        //是clientfd
        else
        {
        i=1;
        //遍历sockarry数组,查找就绪的csock并处理
        for(i;i<4096;i++)
        {
        if(sockarry[i].fd!=-1)
        if(sockarry[i].revents==POLLIN)
        {
        //读取处理数组
        if((len=recv(sockarry[i].fd,buff,sizeof(buff),0))>0)
        {
        buff[strlen(buff)-1]='\0';
        if((strcmp(buff,"time"))==0)
        {
        bzero(timebuff,1024);
        tp=time(NULL);
        ctime_r(&tp,timebuff);
        send(sockarry[i].fd,timebuff,strlen(timebuff),MSG_NOSIGNAL);
        }
         else{
        send(sockarry[i].fd,"try again",10,MSG_NOSIGNAL);
        }
        bzero(buff,1024);
        sockarry[i].revents=0;
        }
        if(len==0)
        {
        //客户端退出,在监听中删除,在组中删除
        close(sockarry[i].fd);
        sockarry[i].fd=-1;
        sockarry[i].revents=0;
        }
        break;
        }
        }
        }
        socknum--;
        }
        }
        close(ssock);
        return 0;
}

用ulimit -a查看一下,如果小于4096,那么创建的sockarray会报错。

epoll模型

整合了select和poll的优势,并且优化了问题

epoll使用红黑树作为监听集合(监听树)

int epfd=epoll_create(int treeMax);//创建监听树,参数为大小,返回值指向树的描述符

struct epolevent node;

node.data.fd=sock;

node.events=EPOLLIN|EPOLLOUT|EPOLLERR;

epoll_ctl(epfd,EPOLL_CTL_ADD,sock,struct epollevent* node);//添加节点

epoll_ctl(epfd,EPOLL_CTL_DEL,sock,NULL);//删除节点

epoll_ctl(epfd,EPOLL_CTL_MOD,sock,&node);//修改只能监听的事件,无法修改sock

struct epollevent ready_arry[Max];//用户空间定义就绪队列(数组)

int readynode=epoll_wait(epfd,ready_arry,Max,-1);//Max就绪最大数,树大小=就绪队列大小=最大就绪数

epoll监听树直接创建于内核,然后节点放入epoll等待队列,然后就绪后弹出到就绪链表(双向链表结构体,内核),然后从内核空间拷贝到用户空间,然后epoll监听到就绪直接返回就绪节点(sock),用户遍历处理这些sock即可。

epoll不用担心轮询问题,所以没有监听限制,可以监听系统最大描述符数量,并没有多余开销

epoll不存在轮询问题,无需担心监听数量增大,系统开销增大。

epoll直接返回就绪的sock,用户直接处理即可。

epoll监听集合在内核层,所以不会出现重复拷贝和重复挂载的问题,保证每个节点只拷贝一次,只挂载一次。

epoll与poll一样,监听的事件更丰富,而后设置监听比较灵活,可以对不同的sock设置不同的事件监听。

作为监听模型,select,poll,epoll,epoll监听能力最强,但是处理能力而言(与监听模型无关)

监听部分(通过io复用技术监听管理有效连接)

处理部分(并发处理),又快又好的将请求全部处理掉(快速反应,提高体验)

epoll的监听模式(水平触发模式EPOLLLT 边缘触发模式EPOLLET)EPOLLONTSHOT

epoll是线程安全的,epoll_ctl函数内部自带互斥锁,所以一个线程在访问修改树节点时,其他线程无法挂起等待

EPOLL+线程池服务器(满足高并发需求)

线程池技术:

预创建原则,线程池内部准备线程备用,不宜过多

线程应该重用性,可以一对多处理任务或服务不同的客户端

处理单元(线程)数量并不固定,动态扩容与缩减(任务量)

设计灵活的任务传递方式与任务接口,线程可以执行不同种类的任务,不能将线程工作固定

线程池技术多为线程容器

thread_poll_create(int max,int min,int qmax);

int thread_producer_add(pool_t* pt,business_t bs);//添加一次业务向队列(生产者)

void* thread_customer(void* arg);//消费者线程任务,等待队列,循环获取业务并执行

void* thread_manager(void* arg);//查看线程池阈值,动态扩容缩减线程

int net_init();//服务器网络初始化

int first_response();//连接成功首次响应

int epoll_init();//epoll创建初始化

int epoll_listen();//主线程循环监听sock事件,并根据就绪添加任务

void* accept_business(void* arg);//tcp连接业务

void* response_business(voiod* arg);//响应处理业务

if_thread_alive(pthread_t tid);//测试线程,返回0表示线程失效,1表示有效

#include<string.h>
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<sys/stat.h>
#include<sys/types.h>
#include<fcntl.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<pthread.h>
#include<sys/epoll.h>
#include<time.h>
#include<signal.h>
#include<errno.h>
#define EPOLL_MAX 190000
int epfd;//监听树的描述符
//business_t业务类型
typedef struct
{
void* (*task)(void* arg);
void* arg;
}business_t;
//pool_t线程池类型
typedef struct
{
//线程池开关
int thread_shutdown;
//线程池最大线程数
int thread_max;
//线程池最小线程数
int thread_min;
//存活有效的线程数量
int thread_alive;
//繁忙线程数量
int thread_busy;
//缩减码
int exit_code;
//环形任务队列
business_t* queue;
int front;
int rear;
int max;
int cur;
//互斥锁
pthread_mutex_t lock;
//生产者条件变量
pthread_cond_t Not_Full;
//消费者条件变量
pthread_cond_t Not_Empty;
//消费者tid数组
pthread_t* ctids;
//管理者线程io
pthread_t mtid;
}pool_t;
pool_t* thread_pool_create(int tmax,int tmin,int qmax);
int thread_producer_add(pool_t* pt,business_t bs);
void* thread_customer(void* arg);
int if_thread_alive(pthread_t tid);
void* thread_mananger(void* arg);
int net_init(void);
int first_response(int sock,struct sockaddr_in caddr);
int epoll_init(int sock);
int epoll_listen(pool_t* pt,int sock);
void* accept_business(void*);
void* recv_business(void*);

#include<pool.h>
void* accept_business(void* arg)
{
        int sock=*(int*)arg;
        int csock;
        struct sockaddr_in caddr;
        char buffer[16];
        bzero(buffer,sizeof(buffer));
        socklen_t addrlen=sizeof(caddr);
        if((csock=accept(sock,(struct sockaddr*)&caddr,&addrlen))==-1)
                perror("accept_business,accept failed");
        first_response(csock,caddr);
        //设置监听
        struct epoll_event node;
        node.data.fd=csock;
        node.events=EPOLLIN|EPOLLET;
        epoll_ctl(epfd,EPOLL_CTL_ADD,csock,&node);
        inet_ntop(AF_INET,&caddr.sin_addr.s_addr,buffer,16);
        printf("customer 0x%x execl accept_business success,client ip %s,client port %d\n",(unsigned int)pthread_self(),buffer,ntohs(caddr.sin_port));
        return NULL;
}

    #include<pool.h>
    int if_thread_alive(pthread_t tid)
    {
    int err;
    err=pthread_kill(tid,0);
    if(err==ESRCH)
    return 0;
    else return 1;
    }

#include<pool.h>
void* recv_business(void* arg)
{
        int sock=*(int*)arg;
        //读取请求数据,使用非阻塞,当前缓冲区无可读数据立即返回
        char buffer[1024];
        char timebuf[1024];
        time_t tp;
        int recvlen;
        bzero(buffer,sizeof(buffer));
        while((recvlen=recv(sock,buffer,sizeof(buffer),MSG_DONTWAIT))>0)
        {
                //检测关键字,处理请求
                buffer[strlen(buffer)-1]='\0';
                if((strcmp(buffer,"time"))==0){
                        bzero(timebuf,1024);
                        tp=time(NULL);
                        ctime_r(&tp,timebuf);
                        send(sock,timebuf,strlen(timebuf),MSG_NOSIGNAL);
                        printf("recv_business execl success,response time\n");
                }
                else{
                        send(sock,"Please try again",17,MSG_NOSIGNAL);
                        printf("customer 0x%x recv_business execl success,reponse tryagain\n",(unsigned int)pthread_self());
                }
        }
        if(recvlen==0){
                //删除监听
                epoll_ctl(epfd,EPOLL_CTL_DEL,sock,NULL);
                printf("client %d exit,close csock\n",sock);
                close(sock);
                printf("customer 0x%xrecv_business execl success, client exit\n",(unsigned int)pthread_self());
        }
        if(recvlen==-1){
                if(errno==EAGAIN){
                        //非阻塞返回
                }
                else perror("recv_bussiness recv call fialed");
        }
 return NULL;
}

#include<pool.h>
extern int epfd;
int epoll_init(int sock)
{
if((epfd=epoll_create(EPOLL_MAX))==-1)
perror("create failed");
struct epoll_event node;
node.data.fd=sock;
node.events=EPOLLIN|EPOLLET;
epoll_ctl(epfd,EPOLL_CTL_ADD,sock,&node);
//创建监听树并将seversock设置成监听读事件
printf("server epoll success\n");
return 0;
}

#include<pool.h>
void* thread_customer(void* arg)
{
        //接收参数
        pool_t* pt=(pool_t*)arg;
        business_t bs;
        pthread_detach(pthread_self());
        //循环等待任务并处理
        printf("customer thread 0x%x,wait business\n",(unsigned int)pthread_self());
        while(pt->thread_shutdown)
        {
                pthread_mutex_lock(&pt->lock);
                while(pt->cur==0)
                {
                        pthread_cond_wait(&pt->Not_Empty,&pt->lock);
                        if(!pt->thread_shutdown){
                                printf("customer 0x%x,thread_shutdown %d,customer exiting..\n",(unsigned int)pthread_self(),pt->thread_shutdown);
                                pthread_mutex_unlock(&pt->lock);
                                pthread_exit(NULL);
                        }
                        if(pt->exit_code>0){
                                printf("manager kill customer 0x%x success\n",(unsigned int)pthread_self());
                                --(pt->thread_alive);
                                --(pt->exit_code);
                                pthread_mutex_unlock(&pt->lock);
                                pthread_exit(NULL);
                        }
                }
                //获取任务
                bs.task=pt->queue[pt->rear].task;
                bs.arg=pt->queue[pt->rear].arg;
                --(pt->cur);
                pt->rear=(pt->rear+1)%pt->max;
                ++(pt->thread_busy);
                pthread_mutex_unlock(&pt->lock);
                //唤醒生产者
                pthread_cond_signal(&pt->Not_Full);
                //执行任务
                bs.task(bs.arg);
                pthread_mutex_lock(&pt->lock);
                --(pt->thread_busy);
                pthread_mutex_unlock(&pt->lock);
        }
        printf("customer 0x%x,thread_shutdown %d,customer exiting..\n",(unsigned int)pthread_self(),pt->thread_shutdown);
        pthread_exit(NULL);
}

#include<pool.h>
void* thread_mananger(void* arg)
{
        pool_t* pt=(pool_t*)arg;
        int alive,busy,cur;
        int flag;
        int add;
        int fd;
        if((access("manager_information",F_OK))==0)
                unlink("manager_information");
        fd=open("manager_information",O_RDWR|O_CREAT,0664);
        char result[4096];
        while(pt->thread_shutdown){
                pthread_mutex_lock(&pt->lock);
                alive=pt->thread_alive;
                busy=pt->thread_busy;
                cur=pt->cur;
                pthread_mutex_unlock(&pt->lock);
                bzero(result,sizeof(result));
                sprintf(result,"manager 0x%x,alive %d ,busy %d,idel %d busy/alive %.2f%% alive/max %.2f%%\n",alive,busy,alive-busy,(double)busy/alive*100,(double)alive/pt->thread_max*100);
                write(fd,result,strlen(result));
                //扩容 thread_min作为扩容与缩减量
                //1.任务数量cur>=alive-busy
                //2.使用百分比扩容,如果忙线程占存活线程的70%
                //3.扩容不能超出最大阈值thread_max
                int err;
                if((cur>=alive-busy||(double)busy/alive*100>=70)&&alive+pt->thread_min<=pt->thread_max)
                {
                        //ctids,如果为0直接使用,非0要测试线程是否存活
                        //遍历tids查找可用位置,并且创建新消费者
                        for(flag=0,add=0;flag<pt->thread_max&&add<pt->thread_min;flag++)
                        {
                                if(pt->ctids[flag]==0||!if_thread_alive(pt->ctids[flag]))
                                {
                                        if((err=pthread_create(&pt->ctids[flag],NULL,thread_customer,(void*)pt))>0)
                                                printf("manager create new customer failed:%s\n",strerror(err));
                                        add++;
                                        pthread_mutex_lock(&pt->lock);
                                        ++(pt->thread_alive);
                                        pthread_mutex_unlock(&pt->lock);
                                 }
                        }
                }
                //缩减条件:闲线程是忙线程的倍数,缩减不允许小于最小阈值
                if(busy*2<=alive-busy&&alive-pt->thread_min>=pt->thread_min)
                {
                        //如何缩减
                        //1.管理者缩减cance,改造ctids,除了存储tid,还要存储记录线程状态,管理者遍历而后cancel取消闲线程
                        //2.配合缩减,消费者退出码,唤醒闲消费者,消费者检查退出码,而后自行退出
                        pt->exit_code=pt->thread_min;
                        int i;
                        for(i=0;i<pt->thread_min;i++)
                                pthread_cond_signal(&pt->Not_Empty);
                }
                sleep(1);
        }
        printf("manager 0x%x exit.\n",(unsigned int)pthread_self());
        pthread_exit(NULL);
}

#include<pool.h>
//生产者监听模型
int epoll_listen(pool_t* pt,int sock)
{
        int readycode;
        struct epoll_event readyarray[EPOLL_MAX];
        int i;
        business_t bs;
        while(pt->thread_shutdown)
        {
                if((readycode=epoll_wait(epfd,readyarray,EPOLL_MAX,-1))==-1)
                        perror("epoll_listen,epoll_wait call failed");
                else{
                        i=0;
                        while(readycode)
                        {
                                //判断就绪
                                if(readyarray[i].data.fd==sock){
                                        //serversock ready
                                        bs.task=accept_business;
                                        bs.arg=(void*)&readyarray[i].data.fd;
                                        thread_producer_add(pt,bs);
                                        printf("thread producer 0x%x add accept business..\n",(unsigned int)pthread_self());
                                }
                                else{
                                        //clientsock ready
                                        bs.task=recv_business;
                                        bs.arg=(void*)&readyarray[i].data.fd;
                                        thread_producer_add(pt,bs);
                                        printf("thrad producer 0x%x add recv business..\n",(unsigned int)pthread_self());
                                }
                                --readycode;
                                ++i;
                        }}
        }
        printf("producer is done..\n");
        return 0;
}

#include<pool.h>
pool_t* thread_pool_create(int tmax,int tmin,int qmax)
{
        //对线程池类初始化,申请空间,按最小阈值创建消费者并创建管理者
        pool_t* pt=NULL;
        if((pt=malloc(sizeof(pool_t)))==NULL)
                perror("thread_pool_create,malloc pool failed");
        pt->thread_shutdown=1;
        pt->thread_max=tmax;
        pt->thread_min=tmin;
        pt->thread_alive=0;
        pt->thread_busy=0;
        pt->exit_code=0;
        //创建队列
        if((pt->queue=malloc(sizeof(business_t)*qmax))==NULL)
                perror("thread_pool_crreate,malloc queue failed");
        pt->front=0;
        pt->rear=0;
        pt->max=qmax;
        pt->cur=0;
        if((pt->ctids=malloc(sizeof(pthread_t)*tmax))==NULL)
                perror("thread_pool_create,malloc ctids failed");
        //初始化ctids,0
        bzero(pt->ctids,sizeof(pthread_t)*tmax);
        if(pthread_mutex_init(&(pt->lock),NULL)!=0||pthread_cond_init(&(pt->Not_Full),NULL)!=0||pthread_cond_init(&(pt->Not_Empty),NULL)!=0){
                printf("thread_pool_create,init lock or cond failed\n");
        }
        //预创建消费者
        int err;
        int i;
        for(i=0;i<tmin;i++){
                if((err=pthread_create(&pt->ctids[i],NULL,thread_customer,pt))>0)
                        printf("thread_pool_create,create customer failed:%s\n",strerror(err));
                else ++(pt->thread_alive);
        }
        //管理线程
        if((err=pthread_create(&pt->mtid,NULL,thread_mananger,pt))>0)
                printf("thread_pool_create,cretate mannager failed:%s\n",strerror(err));
        return pt;
}

#include<pool.h>
int first_response(int sock,struct sockaddr_in caddr)
{
char cip[16];
char response[1500];
bzero(cip,16);
bzero(response,sizeof(response));
inet_ntop(AF_INET,&caddr.sin_addr,cip,16);
sprintf(response,"hi,%s connection server host  success\n",cip);
if((send(sock,response,strlen(response),0))==-1)
perror("first_response,send failed");
return 0;
}

#include<pool.h>
int net_init(void)
{
int sock;
struct sockaddr_in saddr;
bzero(&saddr,sizeof(saddr));
saddr.sin_family=AF_INET;
saddr.sin_port=htons(1234);
saddr.sin_addr.s_addr=htonl(INADDR_ANY);
if((sock=socket(AF_INET,SOCK_STREAM,0))==-1)
perror("sock failed");
if((bind(sock,(struct sockaddr*)&saddr,sizeof(saddr)))==-1)
perror("bind failed");
if((listen(sock,128))==-1)
perror("listen failed");
printf("srver init success\n");
return sock;
}

#include<pool.h>
int thread_producer_add(pool_t* pt,business_t bs)
{
        //生产者任务
        pthread_mutex_lock(&pt->lock);
        if(pt->thread_shutdown)
        {
                while(pt->cur==pt->max)
                {
                        pthread_cond_wait(&pt->Not_Full,&pt->lock);
                        if(!pt->thread_shutdown){
                                printf("cusomter thread 0x%x,shutdown %d,thread exit..\n",(unsigned int)pthread_self(),pt->thread_shutdown);
                                pthread_mutex_unlock(&pt->lock);
                                pthread_exit(NULL);
                        }
                }
                //添加一次业务
                pt->queue[pt->front].task=bs.task;
                pt->queue[pt->front].arg=bs.arg;
                ++(pt->cur);
                pt->front=(pt->front+1)%pt->max;
                pthread_mutex_unlock(&pt->lock);
                pthread_cond_signal(&pt->Not_Empty);

}
pthread_mutex_unlock(&pt->lock);
return 0;
}

#include<pool.h>
int main()
{
        int serverfd=net_init();
        epoll_init(serverfd);
        pool_t* pt=thread_pool_create(100,10,1000);
        epoll_listen(pt,serverfd);
        return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

深入了解:浅拷贝深拷贝

浅拷贝和深拷贝 在java中 理解深拷贝和浅拷贝的概念对于处理对象复制时 保持数据的一致性和独立性至关重要 这两种拷贝方式主要区别在于它们如何处理对象内部的引用类型成员变量 在学习浅拷贝和深拷贝前 我们要区分 对象引用 和 对象本身的区别 对象引用&#xff1a;在Java中…

线上考试系统---虚拟化技术部署

部署前端服务器 1.将资源上传到服务器 scp -r dist/ root192.168.1.50:~2.在服务器上创建基础容器 [rootdocker0 ~]# docker run -it --name c0 centos:latest /bin/bash [rootc1092b8c9ce5 /]# [rootdocker0 ~]# [rootdocker0 ~]# docker attach c0 3.在容器中修改yum源 …

Using Managed Identity to Access Azure OpenAI Service

题意&#xff1a;使用托管身份访问 Azure OpenAI 服务 问题背景&#xff1a; Im working with Azure OpenAI service, and want to test its Managed Identity support. According to the docs it supports Managed Identity authentication, for example - using a VM. 我正…

鸿蒙界面开发(四):支付宝首页开发实战

利用了层叠布局和一个scroll滚动组件。 包括底部导航、顶部导航、主体的滚动窗口。 主体的滚动窗口又包括一个快捷导航栏和一个导航窗口、以及主体部分 import text from ohos.graphics.text;Entry Component struct Index {State message: string Hello World;//支付宝首页b…

全网最最最详细的企业级WEB应用服务器TOMCAT

目录 一:tomcat的介绍 Tomcat请求过程 实验部分&#xff1a; 1&#xff1a;安装 Tomcat 1.1&#xff1a;安装java环境 1.2&#xff1a;安装并启动tomcat 1.3&#xff1a; 查看端口 1.4&#xff1a;浏览器访问以下试试 2&#xff1a;tomcat的文件结构和组成 3&…

高性能web服务器4——Nginx反向代理A

Nginx 反向代理 Nginx 的反向代理功能是其最强大的特性之一&#xff0c;它允许 Nginx 作为中间层来接收客户端的请求&#xff0c;并将这些请求转发到后端服务器。这种架构不仅可以提高安全性&#xff0c;还可以实现负载均衡、缓存和内容过滤等功能。本文将详细介绍 Nginx 反向…

【neo4j】neo4j-Desktop安装

Neo4j是一个高性能的图数据库&#xff0c;它使用图形结构来存储和处理数据。它是一个开源的、完全事务的数据库&#xff0c;专门设计用于大规模的图形数据。Neo4j使用一种名为Cypher的查询语言来处理图形数据&#xff0c;使用户能够方便地进行复杂的图形查询和分析。 Neo4j的主…

AIGC 大模型:实践与未来

一、AIGC 大模型的概念与发展 &#xff08;一&#xff09;AIGC 大模型的定义与特点 AIGC&#xff08;AI Generated Content&#xff09;大模型是一种基于大规模数据训练的人工智能模型&#xff0c;具备强大的内容生成能力和多模态交互能力。它能够理解和处理多种类型的信息&a…

找不到工作四处闲逛github,我发现了一个找工作的神器

前言 到底是谁把我那不值钱的班给上了&#xff0c;博主找不到工作&#xff0c;天天游荡。 终于在github上发现一个找工作的神器&#xff0c;那就是&#xff1a;get_gobs 该项目支持定时投递&#xff0c;自动编写打招呼语&#xff0c;妈妈&#xff0c;再也不用担心我打招呼的…

简易STL实现 | List的实现

基于双向链表的数据结构 1、list的特性 双向链表&#xff1a;允许在序列的两端和中间 执行高效的插入和删除操作 不支持随机访问&#xff1a;要访问list中的元素&#xff0c;必须通过迭代器进行 动态内存管理&#xff1a; list的内部实现使用节点&#xff0c;每个节点都包含…

Java 入门指南:Java IO 模型

UNIX I/O 模型 根据冯.诺依曼结构&#xff0c;计算机结构分为 5 大部分&#xff1a;运算器、控制器、存储器、输入设备、输出设备。 输入设备&#xff08;比如键盘&#xff09;和输出设备&#xff08;比如显示器&#xff09;都属于外部设备。网卡、硬盘这种既可以属于输入设备…

日期反向格式化之前导零

1.问题描述 2.问题分析 为什么用yyyy年MM月dd日会报错&#xff0c;原因是&#xff1a;"前导零" 2.1前导零 前导零指的是在单个数字前面添加一个零以确保数字位数相同的过程。在日期格式化中&#xff0c;前导零常用于确保月份或日期总是显示为两位数字。 例…

mac苹果电脑配置Docker最新国内源

如图&#xff1a; 具体配置如下: {"builder": {"gc": {"defaultKeepStorage": "20GB","enabled": true}},"experimental": false,"registry-mirrors": ["https://docker.anyhub.us.kg", &…

单线程,多线程,异步,同步详解

关于异步与多线程&#xff0c;笔者在刚接触的时候一直存在诸多疑惑&#xff0c;甚至一度以为这俩概念是用来描述同一种技术在不同场景下的应用&#xff0c;进而导致对很多与它们相关的概念都一知半解&#xff0c;代码中的async/await关键词也是莫名其妙地在用。 但是在不断地接…

【解析几何笔记】8.向量的投影与内积

8. 向量的投影与内积 复习前面的知识&#xff1a;&#xff0c;若BCE三点共线&#xff0c;则 A E ⃗ ( 1 − s ) A B ⃗ s A C ⃗ , ( B , C , E ) μ ⇒ s μ 1 μ , 1 − s 1 1 μ \vec{AE}(1-s)\vec{AB}s\vec{AC},(B,C,E)\mu\Rightarrow s\frac{\mu}{1\mu},1-s\frac…

【案例59】WebSphere类加载跟踪开启方法

问题现象 WAS加载代码时&#xff0c;模块开发怀疑是WebSphere本身加载某个类的代码出现了问题。但不知道怎么排查。故寻求帮助。 问题分析 WebSphere本身是提供相关类加载跟踪的方法的。 解决方案 经过排查资料。如果实际诊断中&#xff0c;能够明确断定是某个类的加载出了…

MySQL集群技术详解

目录 一、MySQL在服务器中的部署方法 1.1 编译安装MySQL 1.2 部署MySQL 二、MySQL主从复制 2.1 配置master 2.2 配置slave 2.3 添加slave2 测试&#xff1a; 2.4 延迟复制 2.5 慢查询日志 2.6 MySQL的并行复制 2.7 MySQL主从复制原理剖析 2.8 架构缺陷 三、MySQL…

猫咪掉毛严重,新手铲屎官不知如何处理?推荐使用宠物空气净化器

把小猫接回来一起生活没几天&#xff0c;我就感觉好日子就到头了...猫咪掉毛怎么这么严重啊&#xff0c;我都不敢怎么撸它&#xff0c;一撸满天都是毛&#xff0c;轻轻一搓就是一大团。而且想到还要清理就很头疼&#xff0c;每天都要很多的时间搞卫生。尝试过把它的毛剪短&…

【时时三省】(C语言基础)指针进阶3

山不在高&#xff0c;有仙则名。水不在深&#xff0c;有龙则灵。 ----CSDN 时时三省 一级指针的传参 示例: 这就是一个一级指针传参 思考:当一个函数的参数部分为一级指针的时候&#xff0c;函数能接受什么参数&#xff1f; 二级指针的传参 二级指针示例: pa是一级指针 p…

K8S 1.31 新功能: 跨核分发CPU

​在Kubernetes的最新版本1.31中&#xff0c;一个超酷的新功能&#xff0c;叫做CPUManager的静态策略&#xff0c;里面有个选项叫做distribute-cpus-across-cores。虽然这个功能现在还在测试阶段&#xff0c;也就是alpha版&#xff0c;而且默认是藏起来的&#xff0c;但它的目的…