Linux 网络编程 day5 多路IO转接之改进select and poll
三种多路IO转接方法:select , poll , epoll
改进select多路IO转接,使用数组来保存含有需要连接的套接字cfd,不用循环至1024,节约时间提高效率。
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<ctype.h>
#define SER_PORT 9003
void sys_err(char* s)
{perror(s);exit(1);
}int main(int argc , char *argv[])
{int cfd , lfd , maxi;int client[FD_SETSIZE] ;char buf[BUFSIZ] , str[INET_ADDRSTRLEN];struct sockaddr_in serv_addr , clit_addr;socklen_t clit_addr_len;bzero(&serv_addr , sizeof(serv_addr));serv_addr.sin_family = AF_INET;serv_addr.sin_port = htons(SER_PORT);serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);lfd = socket(AF_INET , SOCK_STREAM , 0);if(lfd == -1)sys_err("socket error");int opt = 1;setsockopt(lfd , SOL_SOCKET , SO_REUSEADDR , (void*)&opt , sizeof(opt));int ret = bind(lfd , (struct sockaddr*)&serv_addr , sizeof(serv_addr));if(ret == -1)sys_err("bind error");listen(lfd , 128);fd_set rset , allset;int maxfd , i , k , n ,j , m , sockfd;maxfd = lfd;maxi = -1;for(j = 0 ; j < 1024 ; j++)client[j] = -1;FD_ZERO(&allset);FD_SET(lfd , &allset);while(1){rset = allset;clit_addr_len = sizeof(clit_addr);ret = select(maxfd + 1 , &rset , NULL , NULL , NULL);if(ret < 0){sys_err("select error");}else if(ret > 0){if(FD_ISSET(lfd , &rset)){cfd = accept(lfd , (struct sockaddr*)&clit_addr , &clit_addr_len);if(cfd == -1)sys_err("accept error");printf("--received from %s at port %d\n" , inet_ntop(AF_INET , &clit_addr.sin_addr.s_addr , str , sizeof(str)) , ntohs(clit_addr.sin_port));for( m = 0 ; m < 1024 ;m++){if(client[m] < 0){client[m] = cfd;break;}}if( m == 1023){printf("too many clients\n");exit(1);}if(m > maxi)maxi = m;FD_SET(cfd , &allset);if(maxfd < cfd)maxfd = cfd ;if(ret == 1)continue;}for(i = lfd + 1 ; i <= maxi ; i++){if((sockfd = client[i]) < 0 )continue;if(FD_ISSET(sockfd , &rset)){n = read(sockfd , buf , sizeof(buf));if(n == 0){close(i);FD_CLR(sockfd , &allset);client[i] = -1;}else if( n > 0){for( k = 0 ; k < n; k++){buf[k] = toupper(buf[k]);}write(STDOUT_FILENO , buf , n);write(sockfd , buf , n);}else{sys_err("read error");}if(ret == 1)break;}}}}close(cfd);return 0 ;
}
poll
int poll(struct pollfd *fds, nfds_t nfds, int timeout);fds:监听的文件描述符数组struct pollfd {int fd; //监听的文件描述符short events; //待监听的文件描述符对应的监听事件取值events : 读POLLIN , 写POLLOUT , 错误POLLERRshort revents; //传入时,给0,满足对应事件会返回非0-->上述取值};nfds:监听数组的实际有效监听个数
timeout:超时时长,毫秒级等待
-1:阻塞等
0:立即返回,不阻塞进程
>0:等待指定毫秒数返回值:成功返回满足对应监听事件的文件描述符总个数初始化
struct pollfd pfds[1024];
pfds[0].fd = lfd;
pfds[0].events = POLLIN;
pfds[0].revents = 0;
read函数返回值
>0:实际读到字节数
=0:socket中,表示对端关闭。close()
<0:如果errno == EINTR 被异常中断,需要重启 。
如果errno == EAGIN/EWOULDBLOCK 以非阻塞方式读数据,但是没有数据。需要再次读
如果errno == ECONNRSET 说明连接被重置 需要close , 移除监听队列。
poll优缺点
优点:
1、自带数组结构,可以将 监听事件 和 返回事件 分离。
2、拓展监听上限。select无法修改,除非重新编译内核。
缺点:
1、不能跨平台。只有Linux和类Unix
2、无法直接定位满足监听事件的文件描述符,编码难度较大
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<ctype.h>
#include<poll.h>
#define SER_PORT 9003void sys_err(char* s)
{perror(s);exit(1);
}int main(int argc , char* argv[])
{int cfd , lfd , sockfd;char buf[BUFSIZ] , str[INET_ADDRSTRLEN];struct sockaddr_in serv_addr , clit_addr ;bzero(&serv_addr , sizeof(serv_addr));socklen_t clit_addr_len;lfd = socket(AF_INET , SOCK_STREAM , 0);if(lfd == -1)sys_err("socket error");int opt = 1 ;setsockopt(lfd , SOL_SOCKET , SO_REUSEADDR , (void*)&opt , sizeof(opt));int ret = bind(lfd , (struct sockaddr*)&serv_addr , sizeof(serv_addr));if(ret == -1)sys_err("bind error");listen(lfd , 128);struct pollfd pfds[1024];int maxi = lfd;pfds[0].fd = lfd;pfds[0].events = POLLIN;int i , n ,k;while(1){ret = poll(pfds , maxi + 1 , -1);if(ret < 0){sys_err("poll error");}else if(ret > 0){if(pfds[0].revents & POLLIN){clit_addr_len = sizeof(clit_addr);cfd = accept(lfd , (struct sockaddr*)&clit_addr , &clit_addr_len);if(cfd == -1)sys_err("accept error");printf("received from %s at PORT %d\n" ,inet_ntop(AF_INET , &clit_addr.sin_addr.s_addr , str , sizeof(str)),ntohs(clit_addr.sin_port));for(i = 0 ; i < 1024 ; i++){if(pfds[i].fd<0){pfds[i].fd = cfd;break;}}if(i == 1024){printf("too many clients\n");exit(1);}if(maxi < i)maxi = i;if(ret == 1)continue;}for(i = 1 ; i <= maxi ; i++){sockfd = pfds[i].fd;if(sockfd < 0)continue;if(pfds[i].revents & POLLIN){n = read(sockfd , buf ,sizeof(buf));if(n == 0){close(sockfd);pfds[i].fd = -1;}else if(n > 0){for(k = 0 ; k < n ; k++){buf[k] = toupper(buf[k]);}write(sockfd , buf , n);write(STDOUT_FILENO , buf , n);}if(ret == 1)break;}}}}return 0 ;
}
突破1024文件描述符限制
cat /proc/sys/fs/file-max -->当前计算机所能打开的最大文件个数,受硬件影响。
ulimit -a -->当前用户下的进程,默认打开文件描述符个数。1024
修改: sudo vi /etc/security/limits.conf 修改soft 基本不用
* soft nofile 65536 ---> 修改默认值,可直接借助命令修改:ulimit -n 数字(注销用户生效)
* hard nofile 100000 ---> 命令修改上限
epoll实现多路IO转接(重点)
本质是一个平衡二叉树,又是其特例红黑树。
epoll_create
创建一颗监听红黑树
int epoll_create(int size);
size:创建的红黑树的监听节点数量(仅供内核参考)。 返回值:成功返回指向新创建的红黑树的根节点的文件描述符fd
失败-1 ,errno
epoll_ctl(重点)
操作监听红黑树
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
epfd:epoll_create函数的返回值
op:对该监听红黑树所做的操作EPOLL_CTL_ADD:添加fd到监听红黑树EPOLL_CTL_MOD:修改fd在监听红黑树的监听事件EPOLL_CTL_DEL:将一个fd从监听红黑树上摘下(取消监听)
fd:待监听的fd
event:本质是struct epoll_event类型的结构体 地址events:EPOLLIN EPOLLOUT EPOLLERRdata:联合体共用体int fd; 对应监听事件的fdvoid *ptr;uint32_t u32;uint32_t u64; 返回值 成功0 , 失败-1 errno
epoll_wait
阻塞监听
int epoll_wait(int epfd, struct epoll_event *events, int maxevents, int timeout);
epfd:epoll_create函数的返回值
events:传出参数【数组】,传出满足监听条件文件描述符结构体
maxevents:数组元素总个数 1024struct epoll_event events[1024];
timeout:超时时长,毫秒级等待
-1:阻塞等
0:立即返回,不阻塞进程
>0:等待指定毫秒数返回值:>0 满足监听的总个数,可以用作循环上限。
=0 , 没有fd满足监听事件
-1 , 失败 errno
流程
lfd = socket();
bind()
listen()int epfd = epoll_create(1024); //epfd,监听红黑树的树根struct epoll_event tep , ep[1024]; //tep用来设置单个fd的属性,ep是epoll_wait函数传出的满足监听事件的数组
tep.events = EPOLLIN;
tep.data.fd = lfd;epoll_ctl(epfd , EPOLL_CTL_ADD , lfd , &tep); //将lfd添加到监听红黑树上while(1){int ret = epoll_wait(epfd , ep , 1024 , -1); //实施监听for(i = 0 ; i < ret ; i++){ if(ep[i].data.fd == lfd){cfd = accept();tep.events = EPOLLIN;tep.data.fd = cfd;epoll_ctl(epfd , EPOLL_CTL_ADD , cfd , &tep);}else{ n = read(); if(n == 0){close();epoll_ctl(epfd , EPOLL_CTL_ADD , cfd , NULL); //摘除最后一个不需要初始化了,直接传NULL。}else if(n>0){小-->大write();}}}
使用epoll进行多路IO转接代码
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<pthread.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<sys/epoll.h>
#include<ctype.h>
#define SER_PORT 9004
void sys_err(char* s)
{perror(s);exit(1);
}int main(int argc , char *argv[])
{int lfd , cfd ,epfd , sockfd , i , n , kfd , k;char buf[BUFSIZ],str[INET_ADDRSTRLEN];struct sockaddr_in serv_addr , clit_addr;socklen_t clit_addr_len;bzero(&serv_addr , sizeof(serv_addr));serv_addr.sin_family = AF_INET ;serv_addr.sin_port = htons(SER_PORT);serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);lfd = socket(AF_INET , SOCK_STREAM , 0);if(lfd == -1)sys_err("socket error");int opt = 1;setsockopt(lfd , SOL_SOCKET , SO_REUSEADDR , (void*)&opt , sizeof(opt));int ret = bind(lfd , (struct sockaddr*)&serv_addr , sizeof(serv_addr));if(ret == -1)sys_err("bind error");listen(lfd , 128);epfd = epoll_create(1024);if(epfd == -1)sys_err("epoll_create error");struct epoll_event tep , ep[1024];tep.events = EPOLLIN;tep.data.fd = lfd;ret = epoll_ctl(epfd , EPOLL_CTL_ADD , lfd , &tep);if(ret == -1)sys_err("epoll_ctl error");while(1){kfd = epoll_wait(epfd , ep , 1024 , -1);if(kfd == -1 ){sys_err("epoll_wait error");}else if(kfd > 0){for(i = 0 ; i < kfd ; i++){sockfd = ep[i].data.fd;if(sockfd == lfd){clit_addr_len = sizeof(clit_addr);cfd = accept(lfd,(struct sockaddr*)&clit_addr, &clit_addr_len);if(cfd == -1)sys_err("accept error");printf("------received from %s at port %d------\n" ,inet_ntop(AF_INET, &clit_addr.sin_addr, str , sizeof(str)),ntohs(clit_addr.sin_port));tep.events = EPOLLIN;tep.data.fd = cfd;ret = epoll_ctl(epfd , EPOLL_CTL_ADD , cfd , &tep);if(ret == -1)sys_err("epoll_ctl error");}else{n = read(sockfd , buf , sizeof(buf));if(n == 0){close(sockfd);epoll_ctl(epfd , EPOLL_CTL_DEL , sockfd , NULL);}else if(n > 0){for(k = 0 ; k < n ; k++){buf[k] = toupper(buf[k]);}write(sockfd , buf , n);write(STDOUT_FILENO , buf , n);}else{sys_err("read error");}}}}}return 0 ;
}
感觉epoll有些东西容易和poll弄混,黑马老师说可以只记epoll,在之后基本都是使用到epoll进行多路IO转接。
epoll实现多路IO转接主要是使用3个函数,第一个函数是epoll_create(1024),参数创建的红黑树的监听节点数量(仅供内核参考)。 第二个函数是epoll_ctl(epfd , EPOLL_CTL_ADD/MOD/DEL , lfd/cfd , &tep) , tep初始化是 struct epoll_event tep ; tep.events = EPOLLIN/OUT/ERR; tep.data.fd = lfd/cfd; 将lfd挂到红黑树上之后进行while循环,进行监听,kfd = epoll_wait(epfd , ep , 1024 , -1); ep是结构体数组,初始化为struct epoll_event ep[1024];
随后只需要根据下标取出epoll_wait函数的传出参数ep数组中的元素进行判断。是否是lfd或者是cfd,如果是lfd说明有要连接的客户端,进行accept连接,若是cfd,则说明有客户端进行写操作,服务器则进行相应的读操作以及小写转大写的操作即可。
相关文章:
Linux 网络编程 day5 多路IO转接之改进select and poll
三种多路IO转接方法:select , poll , epoll 改进select多路IO转接,使用数组来保存含有需要连接的套接字cfd,不用循环至1024,节约时间提高效率。 #include<stdio.h> #include<stdlib.h> #in…...
【iOS】源码阅读(二)——NSObject的alloc源码
文章目录 前言问题发现探索NSObject的alloc源码实现流程探索NSObject为什么直接走objc_alloc,而GGObject先走alloc总结 前言 前面笔者已经学习了alloc相关源码,之前的alloc底层源码实现步骤是以GGObject为基础的,今天我们来探索一下NSObject中…...
如何在短时间内高效复习食品安全员考试?
以下是一些在短时间内高效复习食品安全员考试的方法: 制定科学计划:根据剩余时间和考试内容,将备考时间划分为基础学习、强化巩固和模拟冲刺三个阶段。如基础学习阶段可安排每天学习 2-3 小时,梳理教材知识;强化巩固阶…...
Kotlin空安全解决Android NPE问题
在 Android 开发中,NullPointerException(NPE)一直是最常见的崩溃类型之一。Kotlin 通过创新的空安全机制,在语言层面彻底解决了这一问题。以下是 Kotlin 空安全的核心要点和实战指南: 一、Kotlin 空安全设计哲学 编译期防御:通过类型系统强制区分可空(?)与非空类型显…...
PrimExpr 与 RelayExpr 的区别
PrimExpr 与 RelayExpr 的区别解析 在 TVM 的表达式系统中,PrimExpr 和 RelayExpr 是两种不同层级的表达式类型,分别服务于 TVM 的不同编译阶段和目标场景。以下是它们的核心区别和关联: 1. 设计目标与层级 特性PrimExprRelayExpr所属层级TV…...
R语言助力森林生态研究:从数据处理到群落稳定性分析的完整流程,结合机器学习与案例写作
在生态学研究中,森林生态系统的结构、功能与稳定性是核心研究内容之一。这些方面不仅关系到森林动态变化和物种多样性,还直接影响森林提供的生态服务功能及其应对环境变化的能力。 👉 森林生态系统的结构、功能与稳定性是生态学研究的核心。…...
android-ndk开发(8): ndk 和 clang 版本对照表
android-ndk开发(8): ndk 和 clang 版本对照表 2025/05/06 1. 概要 android-ndk 是基于 clang 的编译工具链。 当 clang 自身的版本变更导致了普通用户的编译、链接报错时, 用户可能只关注到了 ndk 版本, 导致问题的分析浮于表面。 android-ndk 官方…...
《AI大模型应知应会100篇》第50篇:大模型应用的持续集成与部署(CI/CD)实践
第50篇:大模型应用的持续集成与部署(CI/CD)实践 🧾 摘要 在AI大模型开发中,随着模型版本迭代频繁、依赖复杂、部署环境多样,构建一套高效可靠的持续集成与持续交付(CI/CD)流程显得尤…...
Python基于Django的在线考试系统【附源码、文档说明】
博主介绍:✌Java老徐、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专栏推荐订阅👇&…...
windows操作系统开机自启(自动启动) 运行窗口 shell:startup 指令调出开机自启文件夹
打开 “运行” 窗口,输入 shell:startup 把需要开机自启程序的快捷启动方式复制到启动文件夹 (注意:一定要复制快捷启动方式,可以右键启动的文件,发送到桌面快捷方式,如果直接把启动的文件放进去ÿ…...
笔记本外接显示器检测不到hdmi信号
原因:AMD显卡驱动挂了。 其他 异常特征:显示亮度被禁用,无法调整。 修复步骤: ① ②点击更新驱动程序...
论软件的可靠性设计
目录 摘要(300~330字) 正文(2000~2500字,2200字为宜) 背景介绍(500字做左右) 论点论据(1500字做左右)...
【Linux】基础开发工具
L i n u x Linux Linux 环境下的开发工具非常丰富,是程序员和开发人员进行高效开发的必备基础。 L i n u x Linux Linux 环境下的开发工具主要包括 y u m yum yum 软件包管理器、 v i m vim vim 文本编辑器、 g c c / g gcc/g gcc/g 编译器、 g d b gdb gdb 调试工…...
【编程干货】本地用 Ollama + LLaMA 3 实现 Model Context Protocol(MCP)对话服务
模型上下文协议(MCP)本身提供的是一套标准化的通信规则和接口,简化了客户端应用的开发。 MCP 实际上是一套规范,官方把整套协议分成「传输层 协议层 功能层」三大块,并对初始化握手、能力协商、数据/工具暴露、安全…...
华为策略路由
路由策略:是对路由条目进行控制,通告控制路由条目影响报文的转发路径。路由策略为控制平面。 策略路由:是根据报文特征,认为的控制报文从某个即可转发出去,不修改路由表。即策略路由为在转发平面。 路由策略 策略路由…...
Spring Boot3 实现定时任务 每10分钟执行一次,同时要解决分布式的问题 区分不同场景
在Spring Boot 3中实现分布式定时任务,确保多实例环境下任务仅执行一次,可以采用以下方案: 方案一:Redis分布式锁(推荐) import org.springframework.data.redis.core.StringRedisTemplate; import org.sp…...
山东大学项目实训-创新实训-法律文书专家系统-项目报告(四)
项目简介 法律文书专家系统是一个 Web 应用,提供法律文书摘要提取、法律预测报告生成和法律考试问题答疑三大核心功能。用户需要登录或注册后,进入主页面选择所需功能,进行相应的操作。 用户群体 律师:需要快速提取法律文书摘要…...
sqli-labs靶场通关保姆级教学(Get传输篇)Less-1Less-10
sqli-labs靶场通关保姆级教学(Get传输篇)Less-1~Less-10(纯手注) sqli - labs 靶场是一个专门用于网络安全学习和测试 SQL 注入漏洞的开源靶场。包含报错盲注、布尔盲注、基于联合查询的 SQL 注入等多种类型的 SQL 注入漏洞&…...
Day17 聚类算法(K-Means、DBSCAN、层次聚类)
一、聚类算法 1. K-Means 聚类 原理:K-Means 是一种基于划分的聚类算法,目标是将 n n n 个样本划分到 k k k 个簇中,使得簇内样本的相似度尽可能高,簇间样本的相似度尽可能低。算法通过迭代的方式,不断更新簇的质心…...
构建 Web 浏览 AI Agent:Pydantic + MCP 实现指南
在当今快节奏的数字世界中,高效地从网站提取和总结信息可以成为改变游戏规则的利器。大型语言模型(LLM)提供了令人难以置信的能力,但它们本身并不知道如何浏览网络或获取实时内容。 本文演示如何创建一个由Python驱动的AI Agent,它能够阅读和总结网站内容,使其成为研究人员…...
解决 pnpm dev 运行报错的坎坷历程
解决 pnpm dev 运行报错的坎坷历程 在项目开发过程中, 在clone完别人的代码后启动项目时:nodejs 和 pnpm版本都没问题 ,但是 无法运行 pnpm dev 命令启动项目时,往往会遇到各种各样的报错问题。最近在处理 yudao-ui-admin-vue3 项…...
从贴牌到品牌:出海官网如何让中国制造“贵”起来?
在全球经济一体化的当下,中美关税战如同一记重锤,给国际贸易格局带来了巨大震荡。自贸易摩擦爆发以来,双方多次调整关税政策,涉及的商品种类不断增多,税率持续攀升,众多中国企业的出口业务遭受重创…...
ultralytics框架进行RT-DETR目标检测训练
自DETR提出以来,其采用匈牙利匹配方式真正的实现了端到端检测效果,避免了NMS等后处理过程,同时,相较CNN的局部特征提取,其凭借着Transformer强大的全局特征提取能力,在目标检测领域可谓大杀四方,…...
SQLite基本函数
目录 1 核心函数和聚合函数 1.1 核心函数 1.2 聚合函数 2 字符串函数 3 日期和时间函数 4 数学函数 5 JSON函数 (SQLite 3.9.0) 6 窗口函数 (SQLite 3.25.0) 7 加密和安全函数 8 其他实用函数 9 C#代码示例:使用SQLite函数 9.1 准备工作 9.2 代码实现…...
使用Java和LangChain4j实现人工智能:从分类到生成式AI
人工智能(AI)从科幻小说中的梦想逐步演变为现实,驱动了从语音助手到自动驾驶汽车的各种应用。AI 的发展主要基于两种方法:基于编码的传统方法和基于机器学习的现代方法。机器学习通过神经网络和大量训练数据实现分类、生成等任务&…...
数据分析指标体系
目录 1. 构建业务公式,用量化逻辑串联业务 1.1 明确公式结果 1.2 拆解业务过程 1.3 构建计算关系(yaxb) 经典的成交额业务公式 小疑问: 如何让自己的指标看起来更专业? 量化业务过程的量化,到底是什…...
分布式、高并发-Day04
以下是 Day 4 详细学习内容(CAS 与原子操作实战,30 分钟完整计划),包含原理解析、分步代码实战和性能对比: 📖 今日学习目标 掌握 CAS(Compare-And-Swap)无锁算法的核心原理学会使…...
计算机中的逻辑运算
目录 一、总览 二、详情 1. 基本逻辑运算(与、或、非): 2. 其他常用的逻辑运算(异或、同或、与非、或非): 在计算机中,逻辑运算是构成数字电路和计算机程序基础的关键操作。它们处理的是真值…...
Dify - Stable Diffusion
Stable Diffusion 是一种基于文本提示生成图像的工具,Dify 已经实现了访问 Stable Diffusion WebUI API 的接口,因此你可以直接在 Dify 中使用它。以下是在 Dify 中集成 Stable Diffusion 的步骤。 1. 初始化本地环境 推荐使用装有较强 GPU 的机器来安…...
weapp-vite - 微信小程序工具链的另一种选择
weapp-vite - 微信小程序工具链的另一种选择 前言 weapp-vite 是由 笔者 icebreaker 开发的一个基于 vite 的现代化微信小程序开发工具链。我给它设定的目标初心是: 为小程序开发者带来笑容。 自从在 2024 年的 8 月正式发布之后,到现在也过了将近 9 个月的时间。…...
图形化编程重塑 IoT 边缘开发:技术革新与生态竞合新范式
本文以图形化编程技术为核心,深度剖析其在 IoT 边缘开发中的创新应用与行业变革。通过对传统开发困局的系统解构,结合 iVX 项目等典型案例,揭示图形化编程如何通过可视化逻辑设计、自动代码生成及 AI 驱动架构,实现开发效率与应用…...
node-sass安装失败解决方案
1、python环境问题 Error: Cant find Python executable "python", you can set the PYTHON env variable. 提示找不到python2.7版本, 方法一:可安装一个python2.7或引用其他已安装的python2.7 通过设置环境变量可以解决; 方法二&…...
PDF内容搜索--支持跨文件夹多文件、组合词搜索
平时我们接触到的PDF文档特别多,需要对PDF文档做一些处理,那么今天给大家带来的这两个软件非常的棒,可以帮你提升处理文档的效率。 PDF内容搜索 快速检索 我用夸克网盘分享了「PDF搜索PDF 转长图.zip」,点击链接即可保存。打开「…...
我用cursor 搭建了临时邮箱服务-Temp Mail 365
用业余时间搭建了一个临时邮箱,对于后端程序员出身的我,对前端了解的不太多,有了cursor的帮助,补齐了自己的短板,搭建了这个服务,下面对临时邮箱架构设计与安全性做一个分析。 https://temp-mail-365.com 临…...
RN学习笔记 ✅
太无聊了最近,找点事做,学一下RN丰富一下技术栈🫡。但是开发APP除了RN,还有一种选择就是WebView,但是基于WebView的APP的性能被普遍认为不如RN,因为WebView本质上是一个容器,用于在应用中嵌入网…...
使用原生 CSS 实现轮播
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、核心新特性1. ::scroll-button()2. ::scroll-marker()二、基础实现步骤1. 创建滚动容器2. 添加滚动按钮3. 集成滚动标记三、高级功能1. 滚动驱动动画2. 状态查询3. 响应式布局四、展示前言 最新!原生CSS也可以实现轮播…...
C语言进阶—函数(static,递归,回调,指针,内联,变参,结构体尺寸)
目录 一 static函数 1. static变量 1.静态局部变量 2.静态全局变量 2. static函数 二 递归函数 三 指针函数&函数指针 1. 指针函数 2. 函数指针 四 回调函数 五 内联函数 1. 核心特性表 2. 优缺点分析表 3. 用场景建议 3.1 推荐使用场景 3.2 应避免场景 六…...
碰一碰发视频源码搭建的定制化开发实践
在数字化营销与信息交互的浪潮中,碰一碰发视频技术以其便捷性和高效性,成为吸引用户的重要手段。然而,通用的碰一碰发视频系统往往难以满足企业多样化的业务需求。通过对源码进行定制化开发,可以打造出契合特定场景的专属系统。本…...
代码mark:脚本获取包含全角字符的字符串的长度
脚本获取包含全角字符的字符串的长度 function myLen(s) { var r 0; for (var i 0; i < s.length; i) { var c s.charCodeAt(i); // Shift_JIS: 0x0 ~ 0x80, 0xa0 , 0xa1 ~ 0xdf , 0xfd ~ 0xff // Unicode : 0x0 &…...
FPGA----基于ZYNQ 7020实现petalinux并运行一个程序
引言:上一节我们讲到了使用Alinx 7020b自带的sd卡中的petalinux进行epics的编译,但此种方案个性化程度不足。如:我们项目需要FPGA侧的配合,那么我们需要重新编译petalinx。 注意:本文的知识点来自下面两篇文章&#x…...
微服务架构详解
微服务架构的思想本质 我们为什么需要微服务架构,它一定是为了解决我们某些问题才出现了。这篇文章我们讨论下微服务架构模式所解决的问题,带来的挑战,以及他的核心思想本质。 1 早期的服务架构 上图是一个典型的服务分层架构:…...
error:0308010C:digital envelope routines::unsupported
npm run dev 报错: \node_modules\webpack\hot\dev-server.jsnode:internal/crypto/hash:71 this[kHandle] new _Hash(algorithm, xofLen); Error: error:0308010C:digital envelope routines::unsupported opensslErrorStack: [ error:03000086:digital env…...
Blender 初学者指南 以及模型格式怎么下载
glbxz.com glbxz.com 可以直接下载Blender格式模型 第 1 步:打开 这就是 blender 打开时的样子。 您面对的是左侧和右侧的工具栏,顶部是文件作,底部是时间轴,中间是 3D 视图。 Blender 的默认起始网格是一个立方体,…...
开个帖子记录一下自己学spring源码的过程
一、写在开头 简单记录下来时路,以后也能回头看看鼓励下自己。我以前不喜欢拍照,觉得没什么意思,有一天百度云盘给我推不知道什么时候从相册推到百度云相册的照片,那是我口罩时期在家上体育课的照片,我现在回头望过去…...
【coze】工作流(B站视频总结改写)
【coze】工作流(B站视频总结改写) 1.创建智能体2.工作流2.1 获取视频文案2.2 串联试运行2.3 二次创作(大模型)2.4 MD格式2.5 输出2.6 发布并调用 关于工作流: 简单业务, 如果智能体的逻辑比较简单…...
文化符号与隐形的社会话语权力:解码布尔迪厄理论下的意识形态操控机制
文化符号与隐形的社会话语权力:解码布尔迪厄理论下的意识形态操控机制 引言:符号背后的权力博弈 在当代社会,文化符号早已超越其表层的审美或实用功能,成为维系社会等级、塑造意识形态的隐形权力工具。从故宫博物院的金色琉璃瓦…...
一个关于fsaverage bem文件的说明
MNE文档:基于模板 MRI 的 EEG 前向算子 Head model and forward computation 在了解了脑图谱发展的过程之后,对脑的模版有了更深的认识,所以,对于之前使用的正向的溯源文件,进行一下解析,查看包含的信息&a…...
如何解决Kafka集群中Broker磁盘IO瓶颈?
针对Kafka集群Broker磁盘IO瓶颈问题,这里从实际运维场景出发给出解决方案: 1. 分区负载均衡优化 分区迁移策略 # 查看Topic分区分布(识别热点Broker) kafka-topics --bootstrap-server broker1:9092 --describe --topic high_t…...
42 python http之urllib库
作为办公室牛马,日常工作中总少不了和网络数据打交道。比如从公司内部系统抓取数据做报表。Python urllib 库用于操作网页 URL,并对网页的内容进行抓取处理 一、Urllib 基础入门 urllib 是 Python 内置的一个强大的处理 URL 和网络请求的库,它包含了多个模块,每个模块都有…...
如何把阿里云a账号下面的oss迁移到阿里云b账号下面(同区域)
1.登录a账号进入bucket选择同区域复制 2.登录b账号进入bucket选择bucket授权策略,选择接受复制对象,手动输入,然后是a账号的id和角色名字即可。 3.然后在去a账号保存下同步任务,开始同步了就。...