TCP高并发服务器简介(select、poll、epoll实现与区别)

select、poll、epoll三者的实现:

select实现TCP高并发服务器的流程:

  • 一、创建套接字(socket函数):
  • 二、填充服务器的网络信息结构体:
  • 三、套接字和服务器的网络信息结构体进行绑定(bind函数):
  • 四、套接字设置成被动监听(listen函数):
  • 五、创建要监听的文件描述符集合:
  • 使用select函数后,会将没有就绪的文件描述符在集合中去除,所以需要创建两个文件描述符集合,一个是母本read_fds,类似于常量,保持不变,而另一个作为副本read_fds_t,类似于变量,可以改变;
	fd_set read_fds;
    FD_ZERO(&read_fds);
    fd_set read_fds_t;
    FD_ZERO(&read_fds_t);
  • 六、把创建的套接字添加到要监视的集合中:
	FD_SET(sockfd,&read_fds);
    int fd_max = 0;
    fd_max = fd_max > sockfd ? fd_max : sockfd;
  • 七、设置系统时间结构体变量,用来指定超时的时间:
	struct timeval tm_out;
  • 八、等待文件描述符中的事件是否就绪,成功则返回就绪的文件描述符的个数(select函数):
  • select函数:
	#include <sys/select.h>
	int select(int nfds, fd_set *readfds, fd_set *writefds,
                fd_set *exceptfds, struct timeval *timeout);
    /*
	参数:
		nfds:		要监视的最大文件描述符+1
	
		readfds:	要监视的读文件描述符集合 不关心可以传NULL
	
		writefds:	要监视的写文件描述符集合 不关心可以传NULL
	
		exceptfds:	要监视的异常文件描述符集合 不关心可以传NULL
	
		timeout:	超时时间 如果设置成NULL 会一直阻塞 直到有文件描述符就绪
					
	返回值:
	
		成功 就绪的文件描述符的个数
	
		超时 0
	
		失败 -1 重置错误码
	 */
	 				//struct timeval  可以指定超时时间

				    //如果结构体的两个成员都为0 表示非阻塞
				    
				    struct timeval {
				        long    tv_sec;         //秒 
				        long    tv_usec;       //微秒
				    };
				    
		void FD_CLR(int fd, fd_set *set);	//将文件描述符在集合中删除
		
		int  FD_ISSET(int fd, fd_set *set);	//判断文件描述符是否还在集合中
							// 返回0 表示不在了 非0 表示在
		void FD_SET(int fd, fd_set *set);	//向集合中添加一个文件描述符
		
		void FD_ZERO(fd_set *set);			//清空集合

		if(-1 == (ret = select(fd_max + 1,&read_fds_t,NULL,NULL,&tm_out)))
        {
            perror("select error");
            exit(-1);
        }
        else if(0 == ret)
        {
            puts("timeout!!!!!");
            putchar('\n');
            continue;
        }
  • 九、遍历文件描述符集合,判断哪些文件描述符已经准备就绪:
		for(int i = 3; i < fd_max + 1 && 0 != ret; i++)
        {
				...
		}
  • 十、判断文件描述符是否还在集合中,并且接收来自客户端的数据(recv函数)和给客户端发送应答消息(send函数):
            if(FD_ISSET(i,&read_fds_t))
            {
                //说明有新的客户端连接服务器
                if(i == sockfd)
                {   
                    
                    if(-1 == (accept_fd = accept(sockfd,NULL,NULL)))
                    {
                        perror("accept error");
                        exit(-1);
                    
                    }

                    printf("客户端[%d]连接到服务器\n",accept_fd);

                    //将新连接的客户端的套接字添加到要监视的集合中
                    FD_SET(accept_fd,&read_fds);

                    fd_max = fd_max > accept_fd ? fd_max : accept_fd;
                }
                else //之前连接的客户端在向服务器发送信息
                {

                    memset(buf,0,sizeof(buf));
                    if(-1 == (nbytes = recv(i,buf,sizeof(buf),0)))
                    {
                        perror("recv error");
                        exit(-1);
                    }
                    else if(0 == nbytes)
                    {
                        printf("客户端[%d]已断开连接\n",i);

                        //将已断开连接客户端的套接字在文件描述符集合中剔除
                        FD_CLR(i,&read_fds);

                        //关闭套接字
                        close(i);
                        continue;
                    }
                    if(!strncmp(buf,"quit",4))
                    {
                        printf("客户端[%d]已退出\n",i);
                        //将已断开连接客户端的套接字在文件描述符集合中剔除
                        FD_CLR(i,&read_fds);

                        //关闭套接字
                        close(i);
                        continue;
                    }
                    printf("客户端[%d]发来信息[%s]\n",i,buf);

                    //组装应答消息
                    strcat(buf,"----------k");

                    //给客户端发送应答消息
                    if(-1 == send(i,buf,sizeof(buf),0))
                    {
                        perror("send error");
                        exit(-1);
                    }

                }
                ret--; //减少遍历次数
                
            }
  • 十一、关闭套接字(close函数):

poll实现TCP高并发服务器的流程:

  • 一、创建套接字(socket函数):
  • 二、填充服务器的网络信息结构体:
  • 三、套接字和服务器的网络信息结构体进行绑定(bind函数):
  • 四、套接字设置成被动监听(listen函数):
  • 五、创建要监听的文件描述符集合并清空文件描述符集合:
	//创建要监听的文件描述符集合
    struct pollfd new_fds[2048] = {0};
    
    //清空文件描述符集合
    for(int i = 0; i < 2048; ++i)
    {
        new_fds[i].fd = -1;
    }
  • 六、把创建的套接字添加到要监视的集合中:
	FD_SET(sockfd,&read_fds);
    int fd_max = 0;
    fd_max = fd_max > sockfd ? fd_max : sockfd;
  • 七、套接字添加到要监视的集合中,并且设置要监听的事件:
	//套接字添加到要监视的集合中
    new_fds[0].fd = sockfd;

    //设置要监听的事件
    new_fds[0].events |= POLLIN;
  • 八、记录文件描述符集合中最大的文件描述符,并且设置超时的时间:
	//记录文件描述符集合中最大的文件描述符
    int fd_max = 0;
    fd_max = fd_max > sockfd ? fd_max : sockfd;

    //设置超时的时间
    int tm_out = 10000;
  • 九、等待文件描述符中的事件是否就绪,成功则返回就绪的文件描述符的个数(poll函数):
  • poll实现TCP中型并发服务器select实现TCP小型并发服务器区别就是无需每次重置集合,并且可以设置要监视的最大文件描述符的个数,而select至多监视1024个文件描述符
  • poll函数:
	#include <poll.h>
	int poll(struct pollfd *fds, nfds_t nfds, int timeout);
	/*
	参数:
	
		fds:要监视的文件描述符的集合指向自定义的结构体数据
		
		nfds:fds中已经使用的项目的个数
		
		timeout:超时时间单位是毫秒  
		
				设置成10000 表示10s
				
				-1	永久阻塞
				
				0	非阻塞
	返回值:
	
		0		超时
		-1		出错 重置错误码
		正数	成功 返回的就绪的文件描述符的个数
	*/
			struct pollfd {
			   int   fd;         /* 文件描述符 设置成-1 内核就不再监视这一位了*/
			   short events;     /* 要监视的事件 */
			   short revents;    /* 返回的事件 */
			};
			/*
			要监视的事件是用位运算或起来的
			
			要监视的事件放在events字段,而实际就绪的事件在revents字段返回
			
			POLLIN	读事件
			
			POLLOUT	写时间
			
			POLLERR	异常事件
			*/

		if(-1 == (ret = poll(new_fds,fd_max,tm_out)))
        {
            perror("poll error");
            exit(-1);
        }
        else if(0 == ret)
        {
            puts("timeout!!!!!");
            putchar('\n');
            continue;
        }
  • 十、遍历文件描述符集合,判断哪些文件描述符已经准备就绪:
		for(k = 0; k <= fd_max && ret != 0; ++k)
        {   
				...
		}
  • 十一、找到实际就绪的事件的文件描述符,并且接收来自客户端的数据(recv函数)和给客户端发送应答消息(send函数):
           //找到实际就绪的事件的文件描述符
            if(0 != (new_fds[k].revents & POLLIN))
            {
                //说明有新的客户端连接服务器
                if(new_fds[k].fd == sockfd)
                {
                    if(-1 == (accept_fd = accept(sockfd,NULL,NULL)))
                    {
                        perror("accept error");
                        exit(-1);
                    
                    }

                    printf("客户端[%d]连接到服务器\n",accept_fd);

                    //将新连接的客户端的套接字添加到要监视的集合中

                    //遍历文件描述符集合,给新的文件描述符找一个位置
                    for(j = 0; j < 2048; j++)
                    {
                        if(-1 == new_fds[j].fd)
                        {
                            new_fds[j].fd = accept_fd;
                            new_fds[j].events |= POLLIN;

                            fd_max = fd_max > accept_fd ? fd_max : accept_fd;
                            break;
                        }
                    }
                    if(2048 == j)
                    {
                        //此时集合容量满了
                        close(accept_fd);
                    }
                    

                }
                else //之前连接的客户端在向服务器发送信息
                {

                    memset(buf,0,sizeof(buf));
                    if(-1 == (nbytes = recv(new_fds[k].fd,buf,sizeof(buf),0)))
                    {
                        perror("recv error");
                        exit(-1);
                    }
                    else if(0 == nbytes)
                    {
                        printf("客户端[%d]已断开连接\n",new_fds[k].fd);

                        //将已断开连接客户端的套接字在文件描述符集合中剔除
                        close(new_fds[k].fd);
                        new_fds[k].fd = -1;
                        continue;
                    }
                    if(!strncmp(buf,"quit",4))
                    {
                        printf("客户端[%d]已退出\n",new_fds[k].fd);
                        //将已断开连接客户端的套接字在文件描述符集合中剔除
                        close(new_fds[k].fd);
                        new_fds[k].fd = -1;
                        continue;
                    }
                    printf("客户端[%d]发来信息[%s]\n",new_fds[k].fd,buf);

                    //组装应答消息
                    strcat(buf,"----------k");

                    //给客户端发送应答消息
                    if(-1 == send(new_fds[k].fd,buf,sizeof(buf),0))
                    {
                        perror("send error");
                        exit(-1);
                    }

                }
                ret--; //减少遍历次数
            }
  • 十二、关闭套接字(close函数):

epoll实现TCP高并发服务器的流程:

  • 一、创建套接字(socket函数):
  • 通信域选择IPV4网络协议、套接字类型选择流式
	int sock_fd = socket(AF_INET,SOCK_STREAM,0); //通信域选择IPV4、套接字类型选择流式
  • 二、填充服务器和客户机的网络信息结构体:
  • 1.分别定义服务器网络信息结构体变量serveraddr和客户机网络信息结构体变量clientaddr
  • 2.分别求出服务器和客户机的网络信息结构体变量的内存空间大小,以作备用;
  • 3.网络信息结构体清0
  • 4.使用IPV4网络协议AF_INET
  • 5.在终端预留服务器端主机的IP地址inet_addr(argv[1])
  • 6.在终端预留服务器端网络字节序的端口号htons(atoi(argv[2]))
	struct sockaddr_in serveraddr; //定义服务器网络信息结构体变量
	struct sockaddr_in clientaddr;
    socklen_t serveraddr_len = sizeof(serveraddr);//求出服务器结构体变量的内存空间大小
    socklen_t clientaddr_len = sizeof(clientaddr);//求出客户机结构体变量的内存空间大小

    memset(&serveraddr,0,serveraddr_len); //服务器结构体清零
    memset(&clientaddr,0,clientaddr_len);//客户机结构体清零

    serveraddr.sin_family = AF_INET;  //使用IPV4网络协议
    serveraddr.sin_addr.s_addr = inet_addr(argv[1]);  //IP地址
    serveraddr.sin_port = htons(atoi(argv[2]));//网络字节序的端口号
  • 三、设置允许端口复用(setsockopt函数):
  • setsockopt函数:
  • 功能:设置套接字属性;

	#include <sys/types.h>
	#include <sys/socket.h>

	int setsockopt(int sockfd, int level, int optname,
	const void *optval, socklen_t optlen);	
	/*
	参数:
	
		sockfd:套接字
		level:	选项的级别
	
			套接字API级别		SOL_SOCKET
	
			TCP级别			IPPROTO_TCP
	
			IP级别			IPPROTO_IP
	
		optname:选项的名字
	
			套接字API级别
	
				SO_BROADCAST	是否允许发送广播
	
				SO_RCVBUF		接收缓冲区的大小
	
				SO_SNDBUF		发送缓冲区的大小
	
				SO_RCVTIMEO		接收超时时间
					参数使用的是 struct timeval 结构体
					如果超时了 函数调用会立即返回-1
					并将错误码置成 EAGAIN
	
				SO_SNDTIMEO			发送超时时间
	
				SO_REUSEADDR		端口复用
	
			TCP级别
	
				TCP_NODELAY		使能/禁用Nagle算法
	
			IP级别
	
				IP_ADD_MEMBERSHIP	设置加入多播组
	
		optval:	选项的值
	
			没有特殊说明时 使用的都是int类型
	
		optlen:optval的大小
	
	返回值:
	
		成功 	0
	
		失败 	-1 	重置错误码
	*/
  • 特别注意:
  • 使用setsockopt设置允许端口复用时,其在代码的位置在填充网络信息结构体和bind之间;
	int reuse = 1;
    if(-1 == (setsockopt(sock_fd,SOL_SOCKET,SO_REUSEADDR,&reuse,sizeof(reuse))))
    {
        perror("setsockopt error");
        exit(-1);
    }
  • 四、套接字和服务器的网络信息结构体进行绑定(bind函数):

  • 五、套接字设置成被动监听(listen函数):

  • 六、创建红黑树(epoll_create函数):

	#include <sys/epoll.h>
	int epoll_create(int size);
	/*
	功能:
		
		创建epoll/创建epoll实例的描述符
	
	参数:
	
	    size:参数已经被忽略了,只需要填写大于0的值即可
	
	返回值:

        epoll_create 调用成功时会返回一个非负整数epfd,
    
        表示新创建的 epoll 实例的文件描述符,
    
        如果调用失败则返回 -1,并设置 errno 变量以指示具体错误原因
     */
	int epfd = epoll_create(1);
    if(-1 == epfd)
    {
        perror("epoll_create error");
        exit(-1);

    }
  • 七、定义事件结构体变量和存放就绪事件描述符的数组:
  • 事件结构体epoll_event用于描述一个文件描述符上的事件;
			typedef union epoll_data {
               void        *ptr;
               int          fd;  
               uint32_t     u32;
               uint64_t     u64;
           } epoll_data_t;   
           struct epoll_event {
               uint32_t     events;      //EPOLLIN 读 / EPOLLOUT 写
               epoll_data_t data;        //存放用户的数据
           };    

    struct epoll_event event;
    struct epoll_event events[N];
  • 八、将关心的文件描述符加入到红黑树(epoll_ctl函数):
  • 功能:epoll的控制操作或者用于向 epoll 实例中添加、修改、删除事件;
  • epoll_ctl函数:
	int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
	/*
	参数:
	         epfd:epoll的文件描述符
	
	         op:控制方式
	
	            EPOLL_CTL_ADD:添加
	
	            EPOLL_CTL_MOD:修改
	
	            EPOLL_CTL_DEL:删除
	
	         fd:被操作的文件描述符
	
	         event:(事件)结构体指针
	          
	
	返回值:    
				成功返回0,
	
	            失败返回-1 置位错误码
	 */
	//添加要检测事件的描述符
    event.events = EPOLLIN;

    event.data.fd = sock_fd;

    //将关心的文件描述符加入到红黑树
    if(-1 == (epoll_ctl(epfd,EPOLL_CTL_ADD,sock_fd,&event)))
    {
        perror("epoll_ctl error");
        exit(-1);

    }
  • 九、等待文件描述符中的事件是否就绪,成功则返回就绪的文件描述符的个数(epoll_wait函数):
  • epoll_wait函数:
	int epoll_wait(int epfd, struct epoll_event *events,int maxevents, int timeout);
	/*
	参数:

		    epfd:epoll的文件描述符
		
		    events:准备好的事件的结构体地址
		
		    maxevents:返回的最大的文件描述符的个数
		
		    timeout:超时
		
		        >0 :毫秒级别的超时时间
		
		        =0 :立即返回
		
		        =-1:不关心超时时间
	返回值:
	
		     成功返回准备好的文件描述符的个数
		
		     返回0代表超时时间到了
		
		     失败返回-1置位错误码
	*/
		if(-1 == (ret = epoll_wait(epfd,events,N,-1)))
        {
            perror("epoll_wait error");
            exit(-1);
        }	
  • 十、遍历就绪的文件描述符集,判断哪些文件描述符已经准备就绪:
		for(int i = 0; i < ret; ++i)
        {
        	...
        }
  • 十一、找到实际就绪的事件的文件描述符,并且接收来自客户端的数据(recv函数)和给客户端发送应答消息(send函数):
			if(events[i].data.fd == sock_fd)
            {
                //获取连接成功后新的客户端
                new_fd = accept(sock_fd,(struct sockaddr *)&clientaddr,&clientaddr_len);
                if(-1 == new_fd)
                {
                    perror("accept error");
                    exit(-1);
                }
                printf("文件描述符[%d]客户端[%s:%d]连接到了服务器\n",new_fd,inet_ntoa(clientaddr.sin_addr),ntohs(clientaddr.sin_port));
                //添加要检测的文件描述符
                event.events = EPOLLIN;
                event.data.fd = new_fd;
                if(-1 == (epoll_ctl(epfd,EPOLL_CTL_ADD,new_fd,&event)))
                {
                    perror("epoll_ctl error");
                    exit(-1);

                }
                printf("文件描述符[%d]成功挂载在红黑树上\n",new_fd);
            }
            else
            {
                memset(buf,0,sizeof(buf));
                int old_fd = events[i].data.fd;
                if(-1 == (nbytes = recv(old_fd,buf,sizeof(buf),0)))
                {
                    perror("recv error");
                    exit(-1);
                }
                else if(0 == nbytes)
                {
                    printf("文件描述符[%d]客户端断开了服务器\n",old_fd);

                    //关闭对应的文件描述符
                    close(old_fd);
                    //剔除挂在树上对应的文件描述符
                    epoll_ctl(epfd,EPOLL_CTL_DEL,old_fd,&event);

                }
                if(!strncmp(buf,"quit",4))
                {
                    printf("文件描述符[%d]客户端退出了服务器\n",old_fd);
                    //关闭对应的文件描述符
                    close(old_fd);
                    //剔除挂在树上对应的文件描述符
                    epoll_ctl(epfd,EPOLL_CTL_DEL,old_fd,&event);
                }
                printf("文件描述符[%d]客户端发来数据[%s]\n",old_fd,buf);

                //组装应答消息
                strcat(buf,"-----k");

                //给客户端发送应答消息
                send(old_fd,buf,sizeof(buf),0); 
  • 十二、关闭套接字(close函数):

select、poll、epoll三者的区别:

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

14——3

先看一下什么叫转换率的最小值和最大值&#xff0c;看其样例 投入75个o&#xff0c;产出3个x 53个o&#xff0c;换2个x 59个o&#xff0c;换2个x 得出最少20个o换一个x&#xff1b;最多25个o换一个x 也就是说用不同的投入值除以一个相同的数字得到其对应的产出值 而这个相同…

【开源】基于JAVA语言的陕西非物质文化遗产网站

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 设计目标2.2 研究内容2.3 研究方法与过程2.3.1 系统设计2.3.2 查阅文献2.3.3 网站分析2.3.4 网站设计2.3.5 网站实现2.3.6 系统测试与效果分析 三、系统展示四、核心代码4.1 查询民间文学4.2 查询传统音乐4.3 增改传统舞…

阿里云ECS使用docker搭建mysql服务

目录 1.确保正确安装好docker 2.安装mysql镜像 3.创建容器&#xff08;设置端口映射、目录映射&#xff09; 1.确保正确安装好docker 安装教程&#xff1a; 阿里云ECS(CentOS镜像)安装docker-CSDN博客https://blog.csdn.net/qq_62262918/article/details/135686614?spm10…

小白数学建模 Mathtype 7.7傻瓜式下载安装嵌入Word/WPS以及深度使用教程

数学建模Mathtype的下载安装嵌入Word/WPS以及深度使用教程 一 Mathtype 的下载安装1.1 安装前须知1.2 下载压缩包1.3 安装注册 二 嵌入Word/WPS2.1 嵌入Word2.1.1 加载项嵌入 Word2.1.2 宏录制嵌入 Word 2.2 嵌入 WPS2.2.1 加载项嵌入 WPS2.2.2 宏录制嵌入 WPS 2.3 嵌入时报错解…

android 开发 W/TextToSpeech: speak failed: not bound to TTS engine

问题 笔者使用TTS(TextToSpeech)对于文本内容进行语音播报&#xff0c;控制台报错 android 开发 speak failed:not bound to TTS engine详细问题 笔者核心代码&#xff1a; import android.os.Bundle; import android.speech.tts.TextToSpeech; import android.speech.tts.…

GB/T28181-2022之图像抓拍规范解读和设计实现

技术背景 GB/T28181-2022相对2016版&#xff0c;对图像抓拍有了明确的界定&#xff0c;图像抓拍在视频监控行业非常重要, Android平台GB28181设备接入端&#xff0c;无需实时上传音视频实时数据的情况下&#xff0c;就可以抓图上传到指定的图像存储服务器上。 图像抓拍基本要…

Gin 框架之用户密码加密

文章目录 一、引入二、密码加密位置三、如何加密四、bcrypt 库加密4.1 介绍4.2 优点&#xff1a;4.3 使用 五、小黄书密码加密实践 一、引入 Gin是一个用Go语言编写的Web框架&#xff0c;而用户密码的加密通常是在应用程序中处理用户身份验证时的一个重要问题。 通常敏感信息…

军事智能中的深度强化学习不同于传统的深度强化学习

在军事智能中&#xff0c;“诡”和“诈”是两个最重要的概念。 “诡”变指的是智能体通过采取一些不可预测或复杂的变化策略来获得优势。诡变可能包括逃避对手的观察或引诱对手采取不利的行动。智能体可以使用诡变来欺骗对手&#xff0c;使其做出错误的决策或暴露其策略。 “诈…

单表查询练习

目录 题目&#xff1a; 制定约束&#xff1a; 添加表格信息&#xff1a; 所需查询的信息&#xff1a; 实验步骤&#xff1a; 第一步&#xff1a;制作表格 创建新的数据库 创建表格约束&#xff1a; 为表格加入数据&#xff1a; 第二步&#xff1a;查询信息 题目&…

C#,入门教程(21)——命名空间(namespace)与程序结构的基础知识

上一篇&#xff1a; C#&#xff0c;入门教程(20)——列表&#xff08;List&#xff09;的基础知识https://blog.csdn.net/beijinghorn/article/details/124094382 编写软件&#xff08;大软件称为系统&#xff09;与盖大楼一个道理。 假设咱们现在需要盖一座名为“天梯大厦”的…

STM32-调用 vTaskStartScheduler API 后出现 HardFault

STM32 移植 FreeRTOS 后调用 vTaskStartScheduler() 后出现 HardFault 异常。 原因分析&#xff1a; FreeRTOS 配置头文件 FreeRTOSConfig.h 中与中断有关的配置和通过系统接口 void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup) 设置的中断分组冲突。 /* The lo…

含并行连结的网络(GoogLeNet)

目录 1.GoogLeNet 2.代码 1.GoogLeNet inception不改变高宽&#xff0c;只改变通道数。GoogLeNet也大量使用1*1卷积&#xff0c;把它当作全连接用。 V3耗内存比较多&#xff0c;计算比较慢&#xff0c;但是精度比较准确。 2.代码 import torch from torch import nn from t…

Nacos 极简入门

1. 概述 Nacos 是什么&#xff1f;其官方文档自我介绍如下&#xff1a; FROM 什么是 Nacos Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特性集&#xff0c;帮助您快速实现动态服务发现、服务配置、服务元数据及流量管理。 Nacos 帮助您更敏捷和容易…

网易云音乐 API

网易云音乐 API 网易云音乐 API灵感来自环境要求安装运行Vercel 部署操作方法 可以在Node.js调用支持 TypeScript使用文档功能特性更新日志单元测试SDK贡献者License 网易云音乐 API 网易云音乐 Node.js API service 灵感来自 disoul/electron-cloud-music darknessomi/musi…

DDoS攻击规模最大的一次

有史以来DDoS攻击规模最大的是哪一次&#xff1f; Google Cloud团队在2017年9月披露了一次此前未公开的DDoS攻击&#xff0c;其流量达 2.54Tbps&#xff0c;是迄今为止有记录以来最大的DDoS攻击。 在同时发布的另一份报告中&#xff0c;分析高端威胁团体的谷歌安全团队谷歌威胁…

[笔记]深度学习入门 基于Python的理论与实现(四)

4. 神经网络的学习 这里说的‘学习’就是指从训练数据中自动获取最优权重参数的过程。为了进行学习&#xff0c;将导入损失函数这一指标。而学习的目的就是以该损失函数为基准&#xff0c;找出能使它的值达到最小的权重参数。为此&#xff0c;我们介绍利用了函数斜率的梯度法。…

RK3568 android11 移植 v4l2loopback 虚拟摄像头

一&#xff0c;v4l2loopback 简介 v4l2loopback是一个Linux内核模块&#xff0c;它允许用户创建虚拟视频设备。这种虚拟视频设备可以用于各种用途&#xff0c;例如将实际摄像头的视频流复制到虚拟设备上&#xff0c;或者用于视频流的处理和分析等。v4l2loopback的主要作用是创…

大语言模型漏洞缓解指南

虽然大语言模型(LLM)应用正在全球快速普及&#xff0c;但企业对大语言模型的威胁态势仍然缺乏全面了解。面对大语言模型风险的不确定性&#xff0c;企业希望在保障其安全性的基础上加快应用脚步&#xff0c;用人工智能提升企业核心竞争力&#xff0c;这意味着企业的CISO面临着理…

Vray渲染效果图材质参数设置

渲染是创造出引人入胜视觉效果的关键步骤&#xff0c;在视觉艺术领域尤为重要。不过&#xff0c;渲染作为一个资源密集型的过程&#xff0c;每当面对它时&#xff0c;我们往往都会遭遇到时间消耗和资源利用的巨大挑战。幸运的是&#xff0c;有几种方法能够帮助我们优化渲染&…

vue列表飞入效果

效果 实现代码 <template><div><button click"add">添加</button><TransitionGroup name"list" tag"ul"><div class"list-item" v-for"item in items" :key"item.id">{{ i…