做电商网站就业岗位晋升/大数据营销系统怎么样
文章目录
- 1. poll
- 1.1 poll的函数接口
- 1.2 poll的简单测试程序
- 1.3 poll的优缺点分析
- 2. epoll
- 2.1 epoll的函数接口
- 2.2 epoll的工作原理
- 2.3 epoll的工作模式(LT,ET)
- 2.4 epoll的简易服务器实现(默认是LT工作模式)
前言: 接上文讲述的select,它有缺点,而且写起来复杂。于是就有了它的升级版,先出现了poll,到现在的epoll,逐渐进步,写起来方便了不少。poll用的不多,现在基本多路转接都是用的epoll,所以epoll要好好的讲讲,poll就当作选学内容,了解即可。
1. poll
poll与select相比,poll的编写较为简单。因为poll它有一个结构体,去管理关心事件,不需要像select还得自己去写第三方数组。其次poll它没有数量限制,select是数量限制的,但是数量大了必然会影响效率,poll和select在底层都是采用轮询检测的方式去查看要关心的fd的事件是否就绪,像epoll就不会采用这种轮询方式,这个后面讲。
总结:poll作为多路转接的一种方式,它是比select进步了的。
1.1 poll的函数接口
- int poll(struct pollfd *fds, nfds_t nfds, int timeout);
函数参数:
- fds是一个结构体数组,每一个结构体都有它的fd,关心的事件,内核反馈事件。看一下这个结构体:
struct pollfd {
int fd; /* file descriptor /
short events; / requested events /
short revents; / returned events */
};
- events,就是用户告诉内核,它关心fd的事件。
- revents,就是内核告诉用户,你关心的事件怎么样。
- 其实这个events,revents还是位图,事件是宏定义好了的,可以看一下事件:
比如:我要设置结构体关心读事件,可以使得events = POLLIN
,如果还想要关心写事件,那么就是events |= POLLOUT
;怎么判断事件就绪呢?很简单,revents & POLLIN
,为真就表示读事件就绪了。还是一些位操作。
- nfds 代表fds数组的长度.
typedef unsigned long int nfds_t;
注意它的类型 nfds_t 其实就是一个 无符号long int。所谓fds数组长度,可以理解成fds数组中元素个数,可不敢理解成数组大小啊。- timeout是表示poll函数的超时时间, 单位是毫秒(ms),这是一个时间线,在timeout之前是阻塞等待,超过timeout就是非阻塞等待(直接返回)。
比如:你设置为 -1 ,那么就是永远的非阻塞式等待;设置为0,那就是永远的非阻塞式等待;或者你给个合理的时间线,控制一下,都可以。
函数的返回值:
- 返回值小于0, 表示出错;
- 返回值等于0, 表示poll函数等待超时;
- 返回值大于0, 表示poll由于监听的文件描述符就绪而返回.
1.2 poll的简单测试程序
了解函数接口后,就简单的写一个测试程序:
#include <iostream>
#include <poll.h>
#include <unistd.h>
int main()
{struct pollfd rfds;rfds.fd = 0;rfds.events = POLLIN;rfds.revents = 0;while (true){int n = poll(&rfds, 1, -1);switch (n){case -1:std::cerr << "poll errno" << std::endl;break;case 0:std::cout << "time out" << std::endl;default:std::cout << "有事件就绪" << std::endl;if (rfds.revents & POLLIN){std::cout << "读事件就绪" << std::endl;char buffer[1024] = {0};ssize_t s = read(0, buffer, sizeof(buffer) - 1);if (s > 0){std::cout << "Say:" << buffer << std::endl;}}break;}}return 0;
}
可以看到poll的第三个参数我设置的是 -1 ,阻塞式等待。你当然可以改为0,看看现象,肯定是一刻不停的告诉你,time out
。
看一下运行结果:
我就怕小白事后问我,为啥你代码运行时,卡着不动。我告诉你哈,上面的代码检测的文件描述符是0,fd = 0是标准输入流,说人话 就是你 敲键盘 输入的东西,你不敲键盘输入,可不 它就 卡着不动,而且还是阻塞式等待 你输入。
1.3 poll的优缺点分析
- 优点:
- poll的代码简单,select要操作位图,位图还得用专门的函数操作,并且还得自己写第三方数组。poll是用结构体数组,这个结构体就是pollfd,而且结构体里可以有fd信息,关心事件信息,内核反馈信息。非常不错。
- poll没有数量限制。
-缺点:
- poll和select一样,底层采用轮询方式检测就绪fd,所以数量一旦大了,就会效率降低
- poll 也要经常的从用户态到内核态,切换
2. epoll
epoll是对poll的又一进步,epoll它最关键的就是在底层不是用简单的轮询方式检测fd,并且对要监测的fd在底层通过红黑树进行高效的管理。这块在工作原理中详谈。
先看select、poll的缺点:
- 1.每次调用时要重复地从用户态读入参数。
- 2.每次调用时要重复地扫描文件描述符。
- 3.每次在调用开始时,要把当前进程放入各个文件描述符的等待队列。在调用结束后,又把进程从各个等待队列中删除。
总结就是:epoll它克服了这些缺点,被公认为Linux2.6下性能最好的多路I/O就绪通知方法,epoll就是你告诉我要管理哪些事件,然后你就等我通知你 就可以了。
2.1 epoll的函数接口
- int epoll_create(int size);
这个函数就是创建一个epoll模型,它的返回值是一个fd,也就是底层epoll的fd,后续的操作都是在这个epoll模型中,所以它的fd尤为重要。
创建成功返回fd,创建失败返回-1,并设置errno。
至于它的参数 size ,表示的就是epoll模型可以监测的fd的最大值。自从linux2.6.8之后,size参数是被忽略的。
注意
: epoll模型创建成功后,它的返回值是一个fd,所以用完后,要记得close(fd)。
- int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);
这个函数就是用于管理epoll模型的:
函数参数:
- epfd,epoll模型的fd
- op,就是怎么操作,它可以有三个取值,也就是三个宏:
EPOLL_CTL_ADD :注册新的fd到epfd中;
EPOLL_CTL_MOD :修改已经注册的fd的监听事件;
EPOLL_CTL_DEL :从epfd中删除一个fd;- fd,需要监听的fd
- event,这个是事件,告诉epoll模型要监听什么事。
先来看看这个结构体:
struct epoll_event
{
uint32_t events; /* Epoll events /
epoll_data_t data; / User data variable */
} __EPOLL_PACKED;
可以看到,events就是事件,它也是宏:EPOLLIN : 表示对应的文件描述符可以读 (包括对端SOCKET正常关闭);
EPOLLOUT : 表示对应的文件描述符可以写;
EPOLLPRI : 表示对应的文件描述符有紧急的数据可读 (这里应该表示有带外数据到来);
EPOLLERR : 表示对应的文件描述符发生错误;
EPOLLHUP : 表示对应的文件描述符被挂断;
EPOLLET : 将EPOLL设为边缘触发(Edge Triggered)模式, 这是相对于水平触发(Level Triggered)来说的.
EPOLLONESHOT:只监听一次事件, 当监听完这次事件之后, 如果还需要继续监听这个socket的话, 需要再次把这个socket加入到EPOLL队列里.
可以看到epoll_event里面还有一个date,它也是一个结构体,最关键的就是它里面要保存fd。
typedef union epoll_data
{
void *ptr;
int fd;
uint32_t u32;
uint64_t u64;
} epoll_data_t;
- int epoll_wait(int epfd, struct epoll_event *events,int maxevents, int timeout);
这个函数就是用来取出内核反馈事件的:
函数参数:
- epfd:epoll模型的fd
- events:这个结构体组合就是用来存放,内核对监测fd的反馈情况的。
- maxevents:这个参数要小于等于epoll_creat(size)的size,给成size就行。
- timeout:和以往的IO模型一样,这是设置超时时间
返回值:返回值比较有意思,
- 成功:返回对应I/O上已准备好的文件描述符数目,这个写代码的时候能感受到用处
- 失败: 返回 -1,表明函数调用失败
- 超时:返回值等于0
2.2 epoll的工作原理
了解epoll的工作原理,其实就是了解上面的三个函数接口,在底层都干了什么:
- 调用epoll_create创建一个epoll模型;
- 调用epoll_ctl, 将要监控的文件描述符进行注册;
- 调用epoll_wait, 等待文件描述符就绪;
- epoll_create()
调用epoll_create()后在内核会形成一个结构体eventpoll:
/* * This structure is stored inside the "private_data" member of the file * structure and rapresent the main data sructure for the eventpoll * interface. */
struct eventpoll { /* Protect the this structure access,可用于中断上下文 */ spinlock_t lock; /* * This mutex is used to ensure that files are not removed * while epoll is using them. This is held during the event * collection loop, the file cleanup path, the epoll file exit * code and the ctl operations.用户进程上下文中 */ struct mutex mtx; /* Wait queue used by sys_epoll_wait() */ wait_queue_head_t wq; /* Wait queue used by file->poll() */ wait_queue_head_t poll_wait; /* List of ready file descriptors */ struct list_head rdllist; /* RB tree root used to store monitored fd structs */ struct rb_root rbr; /* * This is a single linked list that chains all the "struct epitem" that * happened while transfering ready events to userspace w/out * holding ->lock. */ struct epitem *ovflist; /* The user that created the eventpoll descriptor */ struct user_struct *user;
};
这个结构体维护了epoll模型的基本属性,其中最关键的两个就是:
struct rb_root rbr
和struct list_head rdllist
,这俩结构体大家很熟悉,第一个就是红黑树,第二个就是双向链表。
-
红黑树是用于管理添加进来的事件的,也就是说你创建一个epoll模型,它是用红黑树来保存你向epoll模型中添加的事件。这样好处就是重复添加的事件就可以通过红黑树而高效的识别出来,而且红黑树查找的效率比轮询监测这种好多了。并且用户态调用 epoll_ctl()来操作 epoll 的监视文件时,需要增、删、改、查等动作有着比较高的效率。
-
双向链表是用来保存已经准备就绪事件,以后查询就绪事件,就直接把双向链表的东西拷贝到上层,就是把就绪事件拿到了,并且就绪了n个事件,这个n就是链表的长度,也很方便。
但是有个疑问:谁向双向链表中填入就绪事件呢?是内核直接填嘛?不是,有回调机制。 -
回调机制:所有添加到epoll模型中的事件都会与设备(网卡)驱动程序建立回调方法,也就是说,当响应的事件发生时会调用这个回调方法.这个回调方法在内核中叫ep_poll_callback,它会将发生的事件添加到rdlist双链表中。
-
回调方法:并不是所有的文件都可以用epoll的,它必须有相应的回调方法:
file_operations->poll
,可以看一下:
struct file_operations {ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);__poll_t (*poll) (struct file *, struct poll_table_struct *);int (*open) (struct inode *, struct file *);int (*fsync) (struct file *, loff_t, loff_t, int datasync);// 等
};
看到了把,这个回调方法是和底层硬件定制好了的,它有很多方法。也就是说,事件就绪好了,会通过回调机制(调用ep_poll_callback),去找对应的回调方法,然后再把就绪事件拷贝到双链表中,并且唤醒epoll模型,从而上层调用epoll_wait()可以拷贝双链表的事件。
大的逻辑已经有了,来看图解:
这是epoll模型的大体样子,帮助理解用的。
- epoll_ctl()
epoll_ctl()可以向epoll模型中添加事件,删除事件等操作。
(1) 添加事件:epoll_ctl(int epfd,EPOLL_CTL_ADD , int fd, struct epoll_event *event)
注意op选项设置为EPOLL_CTL_ADD。
首先,它要添加一个事件到epoll模型,这个事件要被管理,用结构体 epitem管理:
struct epitem{ struct rb_node rbn;//红黑树节点 struct list_head rdllink;//双向链表节点 struct epoll_filefd ffd; //事件句柄信息 struct eventpoll *ep; //指向其所属的eventpoll对象 struct epoll_event event; //期待发生的事件类型
}
注意这个epitem是事件在epoll模型中的管理的基本单位,把这个epitem要挂在红黑树里。红黑树的节点是一个键值对,键(key)就是fd,值(value)就是epitem。挂在红黑树上就完成了添加事件。
(2) 删除事件同理,就是红黑树上删除节点呗。
(3)修改事件,就是红黑树的节点的建不变(fd不变),修改它的epitem。
- epoll_wait()
那么等待文件描述符的事情,其实就是等待epoll模型中双向链表中是否有数据,有阻塞等,非阻塞等,设置timeout(三种方式)。但是我们得把基本等待就绪的过程,了解一下,这个就看图解吧:
假如有事件就绪了:
那么epoll_wait()被唤醒了,跑到内核去拷贝head_list中的事件,注意这个函数的返回值是已经准备就绪的事件的个数,就是链表的元素个数,这个效率是O(1)。
head_list中的元素就是事件,事件就是epitem结构体,在强调一遍。
2.3 epoll的工作模式(LT,ET)
工作模式有两种:
- LT:关注的是双链表中是否有
数据
- ET:关注的是双链表中数据是否有
变化
举个例子:
(1)你的快递一共有三个,假如是李四是快递员:
李四:你好,来取下快递。然后你去取快递,你只能拿走两件,并且你拿走了。李四发现你的快递还有遗留,立马通知你:你好,来取下快递。然后你又去取走了快递,这次你取走了最后一个快递。李四发现你的快递没存留,那么也不会通知你了。
(2)你的快递还是有三个,假如王五是快递员:
王五:你好,取一下快递。然后你去取快递,你只能拿走两件,并且你拿走了。但是你还剩下一件,王五不会管,他告诉过你 要来取快递了,取没取完 那是你的事。直到有一天,你又来了一件快递,王五发现你的快递来了,王五:你好,来取快递。你又来取快递,这次你长记性了,快一次就取完吧,不一次取完你都不通知我一下。王五说:只有你的快递从无到有,从少变多时,我才会通知你,换句话说,这有快递数发生变化,我才通知你。
其实李四对应的就是LT工作模式,王五对应的就是ET工作模式。非常有趣昂。
但是要使用ET模式,有一个非常重要的点:fd必须设置为非阻塞模式。
为什么ET模式下,fd必须是非阻塞呢?
当epoll工作在ET模式下时,对于读操作,如果read一次没有读尽buffer中的数据,那么下次将得不到读就绪的通知,造成buffer中已有的数据无机会读出,除非有新的数据再次到达。对吧,这是好理解的。如果到socket编程中,这个问题会造成类似死锁
的情况。
比如:
那么现在就有问题了,epoll_wait()在ET模式下,只有当缓冲区的数据发生变化,才会返回。但是客户端并没有再次发送数据,导致epoll_wait()不返回,一直等待。服务端,它只读取了1k,不完整报文,所有就等epoll_wait()返回,告诉它继续向下拿数据,但问题现在是epoll_wait()不返回。真的无语,现在的情况是,只有当客户端再次发送数据,epoll_wait()才能返回;但是客户端要等服务端的响应,才能继续发送数据。就是这种你等我,我等它,它等你的死循环。
怎么解决这种问题呢?毫无疑问,你必须循环的yici性把10k数据拿走,哪怕你一次拿1k,你循环的拿上10次,必须保证都拿完。这种保证的前提就是 fd是非阻塞的,它可以轮询的拿,而不是等待的拿。
对比LT和ET:
LT是 epoll 的默认行为。 使用 ET 能够减少 epoll 触发的次数, 但是代价就是强逼着程序猿一次响应就绪过程中就把所有的数据都处理完,相当于一个文件描述符就绪之后, 不会反复被提示就绪, 看起来就比 LT 更高效一些.
但是在 LT 情况下如果也能做到每次就绪的文件描述符都立刻处理, 不让这个就绪被重复提示的话, 其实性能也是一样的.
另一方面, ET 的代码复杂程度更高了.
2.4 epoll的简易服务器实现(默认是LT工作模式)
到最后写一个epoll的简易代码,帮助理解。当然它是LT工作模式,如果想要挑战ET模式,那就下点狠功夫,之后我会写一个Reactor的小项目,其中就是利用epoll的ET模式,感兴趣的可以等后续。现在主要是先能把epoll的代码用起来,至于挑战,后面再说。
#include <iostream>
#include <unistd.h>
#include <sys/socket.h>
#include <sys/epoll.h>
#include <netinet/in.h>using namespace std;void useage()
{cout << "Please use"<< "./epoll_server"<< "+"<< "端口号" << endl;
}int main(int argv, char *argc[])
{启动服务器if (argv != 2){useage();exit(1);}使服务器进入listen状态int listen_fd = socket(AF_INET, SOCK_STREAM, 0);if (listen_fd < 0){cerr << "listen failed" << endl;exit(2);}struct sockaddr_in my_sock;my_sock.sin_family = AF_INET;my_sock.sin_addr.s_addr = INADDR_ANY;my_sock.sin_port = htons(atoi(argc[1]));if (bind(listen_fd, (struct sockaddr *)&my_sock, sizeof(my_sock)) < 0){cerr << "bind errno" << endl;exit(3);}if (listen(listen_fd, 5) < 0){cerr << "listen errno" << endl;exit(4);}创建epoll模型,获取epfdint epfd = epoll_create(128); 添加事件,这个就是先得添加listen_fd事件struct epoll_event ev;ev.events = EPOLLIN;ev.data.fd = listen_fd;epoll_ctl(epfd, EPOLL_CTL_ADD, listen_fd, &ev); 事件循环struct epoll_event revs[128]; // 这是用于接收返回fd情况的事件集合while (true){int n = epoll_wait(epfd, revs, 128, -1);switch (n){case -1:cerr << "epoll errno" << endl;break;case 0:cout << "time out" << endl;break;default:// 走到这里说明有事件就绪了cout << "有事件就绪" << endl;for (int i = 0; i < n; i++){int sock = revs[i].data.fd;if (revs[i].events & EPOLLIN){cout << "文件描述符" << sock << "有读事件就绪" << endl;if (sock == listen_fd){cout << "*************************************" << endl;cout << "有连接事件就绪" << endl; 在这里要处理连接事件struct sockaddr_in sockaddr;socklen_t j = sizeof(sockaddr);int fd = accept(listen_fd, (struct sockaddr *)&sockaddr, &j);if (fd < 0){cerr << "accept errno" << endl;}cout << "连接成功" << endl; 连接完成后,要把fd以及它关心的事件,放到epoll模型中struct epoll_event _ev;_ev.data.fd = fd;_ev.events = EPOLLIN;if (epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &_ev) == 0){cout << "成功将" << fd << "托管给epoll" << endl;}else{cerr << "epoll_ctl failed" << endl;}cout << "*************************************" << endl;}else{ 走到这里说明是其他的普通fd的读事件就绪了char buffer[1024] = {0};ssize_t s = read(sock, buffer, sizeof(buffer) - 1);if (s > 0){cout << sock << ":"<< "正常数据读取" << endl;buffer[s] = 0;cout << "client:" << buffer << endl;cout << "*************************************" << endl;}else if (s == 0){ 说明对端连接关闭cout << sock << ":"<< "关闭连接" << endl;close(sock);// 记得管理epoll模型epoll_ctl(epfd, EPOLL_CTL_DEL, sock, NULL);std::cout << "sock: " << sock << "delete from epoll success" << std::endl;cout << "*************************************" << endl;}else{// 读取失败std::cout << "recv error" << std::endl;close(sock);epoll_ctl(epfd, EPOLL_CTL_DEL, sock, nullptr);std::cout << "sock: " << sock << "delete from epoll success" << std::endl;cout << "*************************************" << endl;}}}}break;}}close(epfd);close(listen_fd);return 0;
}
来看现象:
(1)服务器启动,并且客户端1连接:
(2)客户端2也去连接:
客户端使用telnet连接,
服务端现象:
(3)以上说明,连接的epoll管理没有问题,现在客户端1,发送消息:
(4)客户端2发送消息:
(5)以上说明接收数据epoll管理没有问题,那么客户端1退出:
以上测试都没问题,感兴趣的小伙伴,可以下去自己试一试。
相关文章:

IO多路转接 —— poll和epoll
文章目录1. poll1.1 poll的函数接口1.2 poll的简单测试程序1.3 poll的优缺点分析2. epoll2.1 epoll的函数接口2.2 epoll的工作原理2.3 epoll的工作模式(LT,ET)2.4 epoll的简易服务器实现(默认是LT工作模式)前言: 接上文讲述的select,它有缺点,…...

计算机网络整理-问答
1. 程序工作的时候网络各层的状态 如下图所示: 1. TCP 在进行三次握手的时候,IP 层和 MAC 层对应都有什么操作呢? TCP 三次握手是通过在传输层建立连接的一个过程,在这个过程中,TCP 和 IP 层、MAC 层都起到了重要的…...

JS 实现抛物线动画案例
相信大家都有浏览过,很多购物网站购物车的添加商品动画,今天,我们就手写一个简单的抛物线动画,先上案例: 一、绘制页面 我们这里简单实现,一个按钮,一个购物车图标,样式这里直接跳过…...

CSGO搬砖项目,23年最适合小白的项目!
大家好,我是阿阳 不懂的小伙伴,咱继续听我娓娓道来 steam搬砖主要涉及的是csgo游戏平台装备的一个搬运,比较很好理解,主要就是道具的搬运工,简单来讲就是,从国外steam游戏平台购买装备,再挂到…...

谈谈会话管理
客户端和服务器之间进行数据传输遵循的是HTTP协议, 此协议属于无状态协议(一次请求对应一次响应, 响应完之后断开连接), 服务器是无法跟踪客户端的请求, 通过cookie技术可以给客户端添加一个标识, 客户端之后发出的每次请求都会带着这个标识从而让服务器识别此客户端, 但由于co…...

Linux查看JVM FULL GC频率
查看系统的full gc频率,可以使用jstack命令一、采用top命令定位进程登录服务器,执行top命令,查看CPU占用情况,找到进程的pid二、使用jstack命令统计垃圾回收jstat -gc pid 5000即会每5秒一次显示进程号为pid的java进程的GC情况以上…...

java世界String的那些事
String的创建机理: 由于String在Java世界中使用过于频繁,Java为了避免在一个系统中产生大量的String对象,引入了字符串常量池。其运行机制是:创建一个字符串时,首先检查池中是否有值相同的字符串对象,如果…...

【图像配准】多图配准/不同特征提取算法/匹配器比较测试
前言 本文首先完成之前专栏前置博文未完成的多图配准拼接任务,其次对不同特征提取器/匹配器效率进行进一步实验探究。 各类算法原理简述 看到有博文[1]指出,在速度方面SIFT<SURF<BRISK<FREAK<ORB,在对有较大模糊的图像配准时&…...

2023金三银四季跳槽季,啃完这软件测试面试题,跳槽不就稳稳的了
前言 2023年也到来了,接近我们所说的“金三银四”也正在执行了,时间晃眼就过去了,有的人为了2023跳槽早早做足了准备,有的人在临阵磨刀,想必屏幕前的你也想在2023年涨薪吧,那么问题来了,怎么才…...

【C++详解】——vector类
📖 前言:本期介绍vector类。 目录🕒 1. vector的介绍🕒 2. vector的使用🕘 2.1 定义🕘 2.2 iterator🕘 2.3 空间增长🕘 2.4 增删查改🕒 2. vector的模拟实现🕘…...

uniapp 离线本地打包
uniapp打包教程地址 https://nativesupport.dcloud.net.cn/AppDocs/usesdk/android.html点击查看 需要的环境: java (1.8)离线SDK(上面的连接下载即可)Android Studio(同上) 配置环境变量 依次点击“计算机”-“属性”&#…...

初识马尔科夫模型(Markov Model)
初识马尔科夫模型(Markov Model)一、概念二、性质三、学习步骤一、概念 马尔科夫模型(Markov Model)是一种概率模型,用于描述随机系统中随时间变化的概率分布。马尔科夫模型基于马尔科夫假设,即当前状态只…...

CentOS7 ifconfig(或 ip addr)命令不显示IP地址
问题(因为当时没有存图 所以这个图上是网上找的 )解决办法第一:可能是本地服务没有开启,检查本地服务。如图所示,检查这两个服务是否开启。注:如何快速找到服务 可以把光标放在其中一个上面 然后按下VM就可…...

2023/2/10总结
拓扑排序 拓扑排序是在一个有向无环图(DAG)所有顶点的线性排序。 拓扑排序核心思想非常简单,就是先找一个入度为0的顶点输出,再从图中删除该顶点和以它为起点的有向边。继续上面的操作知道所有的顶点访问完为止。 入度…...

2023最新版!宝塔面板Docker自建Bitwarden密码管理
Powered by:NEFU AB-IN 请一定要结合B站视频食用!!!!,下面的博客总体来说只是起到提纲作用 B站视频链接!!! 文章目录2023最新版!宝塔面板Docker自建Bitwarden密码管理前…...

【Hello Linux】 Linux基础命令
作者:小萌新 专栏:Linux 作者简介:大二学生 希望能和大家一起进步! 本篇博客简介:介绍Linux的基础命令 Linux基础命令ls指令lsls -als -dls -ils -sls -lls -nls -Fls -rls -tls -Rls -1总结思维导图pwd指令whoami指令…...

151、【动态规划】leetcode ——2. 01背包问题:二维数组+一维数组(C++版本)
题目描述 原题链接:2. 01背包问题 解题思路 (1)二维dp数组 动态规划五步曲: (1)dp[i][j]的含义: 容量为j时,从物品1-物品i中取物品,可达到的最大价值 (2…...

2023-02-09 - 3 Elasticsearch基础操作
本章主要介绍ES的基础操作,具体包括索引、映射和文档的相关操作。其中,在文档操作中将分别介绍单条操作和批量操作。在生产实践中经常会通过程序对文档进行操作,因此在介绍文档操作时会分别介绍DSL请求形式和Java的高级REST编码形式。 1 索引…...

云原生系列之使用 prometheus监控MySQL实战
文章目录前言一. 实验环境二. 安装MySQL5.72.1 配置yum源2.2 安装MySQL之前的环境检查2.3 开始使用yum安装2.4 启动MySQL并测试三. 安装MySQL_exporter3.1 MySQL_exporter的介绍3.2 mysql_exporter的安装3.3 设置MySQL账户,用于数据收集3.4 启动mysql_exporter3.5 配…...

电脑分盘怎么分?分盘详细教程来了,图文教学
电脑作为小伙伴日常生活使用的工具,很多事情都需要使用电脑来进行处理。虽然小伙伴使用电脑比较多,但是还是有不少的小伙伴不知道电脑分盘怎么分?其实电脑分盘很简单,下面小编就以图文教学的方式,详细的向小伙伴介绍电…...

Element UI框架学习篇(四)
Element UI框架学习篇(四) 1 准备工作 1.0 创建Emp表并插入相应数据的sql语句 /*MySQL数据库*/SET NAMES utf8mb4; SET FOREIGN_KEY_CHECKS 0;-- ---------------------------- -- Table structure for emp -- ---------------------------- DROP TABLE IF EXISTS emp; CRE…...

Revit快速材质切换:同一墙面赋予不同材质的方法
一、Revit中对同一墙面赋予不同材质的方法 方法1:零件法 重点:通过工作平面面板上的设置工作平面命令选取正确的面取消勾选通过原始分类的材质,如图1所示 方法2:拆分构造层绘制一道墙体,选择创建的墙体,单击…...

【Linux operation 56】Linux 系统验证端口连通性
linux 系统验证端口连通性 1、前提 Linux系统有时候需要测试某个端口的连通性,然而ping命令只能测试某个IP通不通,不能测试某端口的连通性。 因为ping命令是基于ICMP协议,是计算机网络中的网络层的协议,但是想要测试某个的连通…...

@Valid注解配合属性校验注解完成参数校验并且优化异常处理
Valid注解配合属性校验注解完成参数校验并且优化参数校验异常处理1 Valid注解配合属性校验注解完成参数校验2 优化参数校验异常处理1 Valid注解配合属性校验注解完成参数校验 向数据库商品分类表中新增商品分类字段,并校验传入的参数 不使用注解的传统方法…...

每天一道大厂SQL题【Day08】
每天一道大厂SQL题【Day08】 大家好,我是Maynor。相信大家和我一样,都有一个大厂梦,作为一名资深大数据选手,深知SQL重要性,接下来我准备用100天时间,基于大数据岗面试中的经典SQL题,以每日1题…...

朗润国际期货:2023/2/10今日期市热点及未来焦点
2023/2/10今日期市热点及未来焦点 1月份人 民币贷款增加4.9万亿元 创历史新高 中国央行: 1月份人民币贷款增加4.9万亿元,同比多增9227亿元。分部门看,住户贷款增加2572亿元,其中,短期贷款增加341亿元,中长期贷款增加…...

TLV73312PQDRVRQ1稳压器TPS622314TDRYRQ1应用原理图
一、TLV73312PQDRVRQ1低压差稳压器 1.2V 300MATLV733 300mA 低压差稳压器是有 300mA 拉电流能力的超小型、低静态电流 LDO,具有良好的线路和负载瞬态性能。这些器件具有 1% 的典型精度。TLV733 系列设计具有先进的无电容器结构,确保无需输入或输出电容器…...

课程回顾|以智能之力,加速媒体生产全自动进程
本文内容整理自「智能媒体生产」系列课程第二讲:视频AI与智能生产制作,由阿里云智能视频云高级技术专家分享视频AI原理,AI辅助媒体生产,音视频智能化能力和底层原理,以及如何利用阿里云现有资源使用音视频AI能力。课程…...

C库函数文件操作(fopen、fread、fwrite、fclose)
C库函数 C文件操作用库函数实现,包含在stdio.h中,系统自动打开和关闭三个标准文件: 标准输入-键盘(stdin)标准输出-显示器(stdout)标准出错输出-显示器(stderr) 文件打…...

【Java|golang】1798. 你能构造出连续值的最大数目
给你一个长度为 n 的整数数组 coins ,它代表你拥有的 n 个硬币。第 i 个硬币的值为 coins[i] 。如果你从这些硬币中选出一部分硬币,它们的和为 x ,那么称,你可以 构造 出 x 。 请返回从 0 开始(包括 0 )&a…...