Linux 实时同步服务实现(Rsync 结合 Inotify)
文章目录
- 1. 实时同步服务介绍
- 2. Inotify 机制介绍
- 3. Inotify-tool+Rsync 实时同步实践
- 3.1 确认远程数据传输服务部署完成
- 3.2 检查Linux系统是否支持Inotify实时监控
- 3.3 安装inotify-tools
- 3.4 命令测试
- 3.5 重要监控事件汇总
- 3.6 使用步骤
- 4. Sersync 工具使用(重点)
- 4.1 Sersync 软件架构及工作原理
- 4.2 Sersync部署环境准备
- 4.3 Sersync程序部署安装
- 5. 解决NFS存储实时同步
Linux 作为主流操作系统,提供了多种强大的实时同步工具。本文主要聚焦 Rsync 结合 Inotify 方式来实现 Linux 实时同步服务,先介绍原理,再通过实践展示配置过程。
1. 实时同步服务介绍
实时同步服务是一种数据处理技术,它能使数据在不同的系统、设备或存储位置之间,以极短的延迟进行更新和保持一致。
- 工作原理
实时同步服务通过持续监控数据源的变化,一旦发现有数据的新增、修改或删除操作,就立即将这些变化传输到目标端,并在目标端应用这些变化,从而确保数据源和目标端的数据在任何时刻都尽可能保持一致。
- 数据备份方案介绍
在网站集群架构中,数据永远是最重要的,一旦数据丢失将会造成巨大损失,因此需要有完善的数据备份方案确保数据安全。
【利用
定时方式实现周期备份
重要数据信息】
需要周期性备份的数据可分为两类:
- 一是程序员发布的程序代码以及运维人员对服务器等的配置变更文件,一般会使用定时任务执行脚本对该类文件进行备份,然后再配合Rsync工具推送到远端服务器备份;
- 二是对于数据库等的数据会用定时执行脚本【通过数据库自带(或第三方)的备份工具定时备份成文件】备份,然后再配合Rsync 工具推送到远端服务器备份。
【利用
实时复制方式
实现实时备份重要数据信息】
实时复制是实际工作中数据备份最重要的方式,主要用于用户提交的数据的备份。
- 对于用户提交到服务器上的普通文件(图片、压缩包、视频、文档等),可采用
Inotify/Sersync + Rsync
实时备份方式;- 对于数据文件还有较复杂的分布式存储工具可以实现将数据同时备份成多份,如
FastDFS
、Synchronize
、GlusterFS
等;- 对于提交到数据库中的数据,可使用数据库的
主从复制
(如 MySQL、Oracle)方式备份,这是软件自带的实时复制备份方式。
这里主要对
Inotify + Rsync
模式的实时同步实现方式进行介绍。
2. Inotify 机制介绍
Inotify 是 Linux 内核提供的一种异步的文件系统变化通知机制,它允许应用程序通过内核接口监视文件系统的变化,从而实现对文件或目录的实时监控,能够监控文件系统中添加、删除、修改、移动等各种事件,使得第三方软件可以监控文件系统下文件的各种变化情况。
- Inotify 机制工作原理
Inotify机制的工作原理如下:
- 初始化:应用程序使用
inotify_init
系统调用来创建一个inotify实例,内核会为该实例分配一个独立的排序队列,并返回一个文件描述符,用于后续对该实例的操作。 - 添加监控:应用程序通过
inotify_add_watch
系统调用,为指定的文件或目录添加监视描述符。在这个过程中,应用程序需要指定要监听的事件类型,如文件的创建、删除、修改等。内核会为每个监视的目标创建一个inotify_watch
结构,并将其插入到被监视目标对应的inode
的inotify_watches
列表中,以此来跟踪和管理监视的文件或目录以及感兴趣的事件。 - 事件产生与处理:当被监视的文件或目录发生变化时,例如进行了读写、创建、删除等操作,相应的文件系统代码会调用
fsnotify_*
系列函数(*号代表具体的事件名),最终调用inotify_inode_queue_event
函数。该函数会遍历inotify_watches
列表,判断当前的文件操作事件是否被某个watch
所监视。如果是,则进一步调用inotify_dev_queue_event
函数。inotify_dev_queue_event
函数会先判断该事件是否为上一个事件的重复,如果是则丢弃;否则,检查inotify
实例的事件队列是否溢出,如果溢出则产生一个溢出事件,否则创建一个当前文件操作事件,将其插入到inotify_device
对象的events
事件列表中,并唤醒等待在inotify_device
结构中wq
指向的等待队列中的进程。 - 读取通知:应用程序使用
read
系统调用,通过inotify_init
返回的文件描述符来读取事件通知。一次read
调用可以获取多个事件,只要提供的缓冲区足够大。应用程序可以根据读取到的事件信息来执行相应的操作。 - 移除监控:应用程序使用
inotify_rm_watch
系统调用来停止对某个文件或目录的监视,内核会释放相应的inotify_watch
结构。当应用程序不再需要使用inotify
实例时,关闭inotify_init
返回的文件描述符,内核会释放inotify_device
结构以及相关的资源。
【通俗一点来讲】
Inotify 机制就像是一个 “文件系统小管家”,专门负责留意文件和文件夹的各种变化,并及时把这些变化告诉需要知道的程序。
- 建立联系
当一个程序想要知道文件系统里发生了什么变化时,它会向操作系统说:“我要关注文件变化啦!” 这时,操作系统就会创建一个类似 “情报站” 的东西,还会给程序一个 “通行证”(文件描述符),程序以后就用这个 “通行证” 和 “情报站” 交流。- 指定关注对象
程序拿到 “通行证” 后,就会告诉 “情报站”:“我想关注这些文件和文件夹,它们要是有创建、删除、修改这些情况,你就赶紧告诉我。” “情报站” 会把这些要关注的目标记下来,就像在本子上列了个清单- 发现变化并报告
在程序关注的文件和文件夹里,只要有任何风吹草动,比如新建了一个文件、删除了一个文件夹或者修改了某个文件的内容,“情报站” 安插在这些地方的 “小眼线” 就会察觉到。“小眼线” 会把这些变化信息整理成 “情报”,放到 “情报站” 的 “收件箱” 里。同时,“情报站” 还会提醒程序:“有新情报啦,快来看看!”- 程序获取信息
程序收到 “情报站” 的提醒后,就会用 “通行证” 去 “情报站” 的 “收件箱” 里取 “情报”。程序会根据这些 “情报” 的内容,决定下一步要做什么,比如更新自己的数据、备份文件等。- 结束关注如果程序不想再关注某些文件或文件夹了,它会告诉 “情报站”:“这些不用再盯着了。” “情报站” 就会把这些目标从清单上划掉。要是程序完全不需要 “情报站” 服务了,它就会交回 “通行证”,“情报站” 也会把自己关闭,把相关的资源都释放掉。
Inotify程序原理是一种事件驱动机制,它为应用程序监控文件系统事件提供了实时响应事件的机制,可以做到对事件处理的实时响应。
这里我们详细说明的主要是
inotify-tools
和sersync
两种实现方式。接下来我们结合实践来看看实时同步如何实现。
3. Inotify-tool+Rsync 实时同步实践
实时同步软件核心功能是监控指定的目录内的数据变化。实现复制到远端服务器依然需要Rsync工具的配合,这些软件几乎都是在Inotify机制的接口上的软件封装,具体的工作机制流程为:
- 备份源客户端开启运行Inotify软件服务进程,监测指定目录的文件系统变化;
- 一旦获取到指定监控目录的数据发生变化,即执行rsync命令复制数据;
- 将变化的数据发送到rsync服务端的备份目录中。
因为之前的博文里已经有说明Rsync服务的搭建方法,所以接下来的Inotify 配置是默认在rsync服务搭建完成的。
3.1 确认远程数据传输服务部署完成
前提是Rsync服务配置成功,可以在客户端上推送、拉取数据到BACKUP备份服务器,然后才能配置Inotify-tools
工具服务。客户端需要做如下部署:
# 采用密码环境变量的方式,配置永久生效。
[root@nfs ~]# echo 'export RSYNC_PASSWORD=123456' >>/etc/bashrc
[root@nfs ~]# source /etc/bashrc
[root@nfs ~]# echo $RSYNC_PASSWORD
123456## 测试推送和拉取数据信息
[root@nfs ~]# rsync -avz /data rsync_backup@172.16.1.41::backup/
sending incremental file listsent 164 bytes received 25 bytes 126.00 bytes/sec
total size is 0 speedup is 0.00
3.2 检查Linux系统是否支持Inotify实时监控
根据官方说明,内核从 2.6.13
起,才支持Inotify功能,因此需要在部署实时复制服务之前,查看系统的版本以及具体的查看是否支持。
通过检查
/proc/sys/fs/inotify
目录中是否存在以下三个文件,可以确认系统是否支持实时监控程序。
[root@nfs ~]# uname -r
3.10.0-957.5.1.el7.x86_64 《== 确认内核支持机制
[root@nfs ~]# ls -l /proc/sys/fs/inotify/
总用量 0
-rw-r--r-- 1 root root 0 4月 19 09:45 max_queued_events
-rw-r--r-- 1 root root 0 4月 19 09:45 max_user_instances
-rw-r--r-- 1 root root 0 4月 19 09:45 max_user_watches## 显示这三个文件证明系统支持Inotify程序功能
文件名称 | 作用说明 |
---|---|
max_user_watches | 设置inotifywait 或 inotifywatch 命令可以监视的文件数量(单线程) |
max_user_instances | 设置每个用户可以运行的 inotifywait 或 inotifywatch 命令的进程数 |
max_queued_events | 设置Inotify 实例事件(event)队列可容纳的事件数量。实例的概念就是,多个相同的服务,启动运行多个进程。 |
/proc/sys/fs/inotify
目录中的这三个文件可以用来限制通过Inotify接口消耗内核内存的数量。默认配置内容:
[root@nfs ~]# cat /proc/sys/fs/inotify/max_user_watches 8192 [root@nfs ~]# cat /proc/sys/fs/inotify/max_user_instances 128 [root@nfs ~]# cat /proc/sys/fs/inotify/max_queued_events 16384
3.3 安装inotify-tools
# 安装epel源,默认的官方源没有 inotify-tools 工具包。
yum install epel-release -y # 下载Inotify软件工具
yum install inotify-tools -y# 查看软件命令
[root@nfs ~]# rpm -ql inotify-tools|head -2
/usr/bin/inotifywait # 软件命令,对指定的文件或目录进行监控
/usr/bin/inotifywatch # 软件命令,收集文件系统事件的统计信息## 查看软件版本
[root@nfs ~]# rpm -qa inotify-tools
inotify-tools-3.14-8.el7.x86_64
安装 inotify-tools
软件之后,主要通过两个命令来实现数据信息的监控。
- 一个是利用
inotifywait
命令实现对数据变化事件的监控。在被监控的目录等待特定文件系统事件(open、close、delete等)发生,执行后处于阻塞状态,适合在shell脚本中使用。(此命令是重点) - 另一个是利用
inotifywatch
命令实现对数据时间信息的统计。收集被监控的文件系统使用的统计数据,指文件系统事件发生的次数。
【
inotifywait
命令参数】
命令参数 参数说明 -m | --monitor 始终保持事件监听状态(重要参数) -d | --daemon 类似于 -m 参数,只是将命令运行在后台
记录触发的事件信息在指定的文件中,利用--outfile
参数
定义程序日志使用--syslog
参数-r 递归监控目录数据信息变化(重要参数) -o | --outfile 打印事件到文件中,相当于标准正确输出 -s | --syslog 发送错误到syslog,相当于标准错误输出 -q | --quite 输出信息少(只打印事件信息) –excludei 排除文件或目录时,不区分大小写 –timefmt 指定时间输出格式 –format 打印使用指定的输出,类似格式字符串,即实际监控输出的内容 -e 指定监听指定的事件,如果省略,则表示所有事件都进行监听。(重要参数) 以上参数主要为命令相关参数,利用Inotify软件主要对数据产生的事件进行实时监控,下面表格列出来了常用的监控时间的信息:
事件名称 事件说明 access 文件或目录内容被读取 modify 文件或目录内容被写入 attrib 文件或目录属性改变 close_write 文件或目录关闭,在写入模式打开之后关闭的 close_nowrite 文件或目录关闭,在只读模式打开之后关闭的 close 文件或目录关闭,不管是读或是写模式 open 文件或目录被打开 moved_to 文件或目录被移动到监控的目录中 moved_from 文件或目录从监控的目录中被移动 move 文件或目录只要发生移动就触发事件 create 文件或目录被创建在监控的目录中 delete 文件或目录被删除,在监控的目录中 delete_self 文件或目录被删除 unmount 文件系统包含的文件或目录不能被卸载
3.4 命令测试
## 命令格式:
【命令】 【命令参数相应】 【监控事件】 【监控目录】
inotifywait -mrq --timefmt '%d/%m/%y %H:%M' --format '%T %w%f' -e create /data# 监控 /data 目录创建事件显示信息
# 命令执行之后,在/data目录下文件会随着文件创建实时出现。
# 此命令只能监控创建事件。
3.5 重要监控事件汇总
在实际使用时,只要监控以下事件即可:create(创建)、delete(删除)、moved_to(移入)、close_write(修改)
3.6 使用步骤
- 第一步:手动测试语句
[root@nfs ~]# inotifywait -mrq --format '%w%f' -e close_write,delete /data
## 同时监控/data目录的增删改事件
- 第二步:编写实时监控和复制的脚本
[root@nfs scripts]# cat monitor.sh
#!/bin/sh
cmd="/usr/bin/inotifywait"
$cmd -mrq --format '%w%f' -e close_write,delete /data|\
while read line
do # 《== 删除事件发生复制动作,用rsync命令对整个目录复制[ ! -e "$line" ] && cd /data &&\ #《== 删除文件时,因为文件已经不存在了,因此只能切换到复制目录下。rsync -az --delete ./ rsync_backup@172.16.1.41::backup && continue#《== 对整个目录进行复制,然后终止当前循环#《== 处理增改事件复制动作,只针对修改的文件进行复制,提升性能rsync -az --delete $line rsync_backup@172.16.1.41::backup
done
- 第三步:测试效果,没问题的话,加入
rc.local
,实现开机自启动。
Sersync
相比 inotify - tools
优势显著。它将监控与同步集成,以多线程实现高效实时同步,无需复杂脚本。通过 XML 配置文件管理,参数丰富且灵活。具备失败重试机制,日志记录详尽助于监控,资源管理优化,稳定性高 。
4. Sersync 工具使用(重点)
Sersync
项目利用Inotify+Rsync
技术实现对服务器数据实时复制的解决方案,其中Inotify
用于监控Sersync
所在服务器上文件系统的事件变化,当事件发生变化时就调用Rsync
命令将变化的数据复制到远端服务器上。
4.1 Sersync 软件架构及工作原理
Inotify
监控指定目录对应事件的变化,当有事件变化时进入事件过滤队列。- 过滤队列负责过滤不需要复制的数据,也可以过滤短时间内产生的
重复Inotify事件
信息,过滤后的事件触发Rsync
对变化数据进行复制。 - 图中的线程组线程是等待线程队列的守护线程,当事件队列中有事件发生的时候,线程组守护线程就会逐个唤醒复制线程,当队列中的
Inotify
事件较多的时候,复制线程就会被全部唤醒一起工作,以提升复制的效率。 - 除了线程组线程之外,还有
Sersync
服务线程负责处理复制失败的文件,将他们再次复制,对于再次复制失败的文件(命令)记录到rsync_fail_log.sh
脚本,然后定期再次执行脚本,同时利用自带的cron
功能,实现每隔一定时间将所有未复制的数据再次执行复制。
4.2 Sersync部署环境准备
同上3.1和3.2的内容,这里不再进行重复。
4.3 Sersync程序部署安装
此处的部署是按照解决nfs单点故障的解决来进行部署的。可以根据实际情况灵活调整。
在NFS服务器上部署Sersync服务,配置sersync实时监控共享目录
/data
的变化,将发生的变化实时推送到rsync备份服务器。
- 安装 inotify-tools
[root@nfs ~]# yum -y install inotify-tools
- 下载或者windows上传sersync压缩包,建议上传,不要直接下载
[root@nfs ~]# wget https://raw.githubusercontent.com/wsgzao/sersync/master/sersync2.5.4_64bit_binary_stable_final.tar.gz
[root@nfs ~]# ls
sersync2.5.4_64bit_binary_stable_final.tar.gz
[root@nfs ~]# mkdir /server/tools -p
# 上传到/server/tools目录
[root@nfs tools]# mv /root/sersync2.5.4_64bit_binary_stable_final.tar.gz sersync.tar.gz
[root@nfs tools]# ll
total 1936
-rw-r--r-- 1 root root 1981010 Dec 6 09:25 sersync.tar.gz
- 解压目录
[root@nfs tools]# tar xf sersync.tar.gz
# 改一下文件夹名
[root@nfs tools]# mv GNU-Linux-x86 sersync
[root@nfs tools]# ll
total 0
drwxr-xr-x 2 root root 41 Oct 26 2011 sersync
[root@nfs tools]# cd sersync/
[root@nfs sersync]# ll
total 1772
-rwxr-xr-x 1 root root 2214 Oct 26 2011 confxml.xml
-rwxr-xr-x 1 root root 1810128 Oct 26 2011 sersync2
- 配置文件
confxml.xml
<?xml version="1.0" encoding="ISO-8859-1"?>
<!--软件版本-->
<head version="2.5"><!--主机IP和端口号--><host hostip="localhost" port="8008"></host><!--是否开启排错--><debug start="false"/><!--文件系统--><fileSystem xfs="false"/><!--过滤功能,排除指定的文件不进行复制--><filter start="false"><exclude expression="(.*)\.svn"></exclude><exclude expression="(.*)\.gz"></exclude><exclude expression="^info/*"></exclude><exclude expression="^static/*"></exclude></filter><!--软件监控指定的事件信息:表示触发监控管理的事件--><inotify><delete start="true"/><createFolder start="true"/><createFile start="true"/><closeWrite start="true"/><moveFrom start="true"/><moveTo start="true"/><attrib start="false"/><modify start="false"/></inotify><!--Sersync主要参数信息配置--><sersync><!--被监控的目录--><localpath watch="/data"><!--指定复制到的机器(Rsync服务器),IP和模块名。支持多台机器,多模块复制--><remote ip="172.16.1.41" name="data"/><!--<remote ip="192.168.8.39" name="tongbu"/>--><!--<remote ip="192.168.8.40" name="tongbu"/>--></localpath><!--配置Rsync服务参数信息--><rsync><!--rsync命令的参数--><commonParams params="-avz"/><!--匿名(认证)用户和密码文件--><auth start="true" users="rsync_backup" passwordfile="/etc/rsync.password"/><!--端口,支持修改--><userDefinedPort start="false" port="874"/><!-- port=874 --><!--超时时间--><timeout start="true" time="100"/><!-- timeout=100 --><!--支持SSH远程--><ssh start="false"/></rsync><!--传输失败内容放置的文件,默认每60分钟执行一次--><failLog path="/tmp/rsync_fail_log.sh" timeToExecute="60"/><!--default every 60mins execute once--><!--定时任务配置--><crontab start="false" schedule="600"><!--600mins--><!--定时任务排除复制的内容--><crontabfilter start="false"><exclude expression="*.php"></exclude><exclude expression="info/*"></exclude></crontabfilter></crontab><!--是否使用其他的插件--><plugin start="false" name="command"/></sersync><!--插件配置内容--><plugin name="command"> <!--命令部分配置--><param prefix="/bin/sh" suffix="" ignoreError="true"/> <!--prefix /opt/tongbu/mmm.sh suffix--><filter start="false"><include expression="(.*)\.php"/><include expression="(.*)\.sh"/></filter></plugin><plugin name="socket"> <!--socket部分配置--><localpath watch="/opt/tongbu"><deshost ip="192.168.138.20" port="8009"/></localpath></plugin><plugin name="refreshCDN"> <!--刷新CDN部分配置--><localpath watch="/data0/htdocs/cms.xoyo.com/site/"><cdninfo domainname="ccms.chinacache.com" port="80" username="xxxx" passwd="xxxx"/><sendurl base="http://pic.xoyo.com/cms"/><regexurl regex="false" match="cms.xoyo.com/site([/a-zA-Z0-9]*).xoyo.com/images"/></localpath></plugin>
</head>
- 启动服务
# 命令 参数 文件
[root@nfs ~]# /server/tools/sersync/sersync2 -dro /server/tools/sersync/confxml.xml
# 或者使用相对路径
[root@nfs sersync]# ./sersync2 -dro ./confxml.xml
- 查看帮助
[root@nfs sersync]# ./sersync2 -h # 或者使用绝对路径/server/tools/sersync/sersync2
set the system param
execute:echo 50000000 > /proc/sys/fs/inotify/max_user_watches
execute:echo 327679 > /proc/sys/fs/inotify/max_queued_events
parse the command param
_______________________________________________________
参数-d:启用守护进程模式
参数-r:在监控前,将监控目录与远程主机用rsync命令推送一遍
参数-n: 指定开启守护线程的数量,默认为10个
参数-o:指定配置文件,默认使用confxml.xml文件
参数-m:单独启用其他模块,使用 -m refreshCDN 开启刷新CDN模块
参数-m:单独启用其他模块,使用 -m socket 开启socket模块
参数-m:单独启用其他模块,使用 -m http 开启http模块
不加-m参数,则默认执行同步程序
________________________________________________________________
sersync参数 | 参数解释说明 |
---|---|
-d | 启用守护进程模式(重要参数) |
-r | 在监控前,将监控目录与远程主机用rsync命令推送一遍 |
-n | 指定开启守护线程的数量,默认为10个 |
-o | 指定配置文件,默认使用confxml.xml文件 |
-m | ①、单独启用其他模块,使用-m refreshCDN 开启刷新CDN模块 ②、单独启用其他模块,使用-m socket 开启socket模块 ③、单独启用其他模块,使用-m http 开启 HTTP 模块如果不加 -m 的话,则默认执行复制程序。 |
注意事项:参数出错虽然启动运行了,但是不推送,不监控……
出问题时,进行调整配置文件后,往往会多次调整,启动服务。这样会导致许多进程在执行,所以需要找到多余的存活进程,干掉进程。
1.过滤进程
[root@nfs ~]# ps axu|grep sersync
2.结束进程后再修改,修改完成在运行
Kill -9 pid
5. 解决NFS存储实时同步
有些时候觉得NFS性能不高,其实在共享存储的时候,可以使用读写分离的方案。
-
用户上传时把文件上传到NFS上,这部分写的并发要比读小的多,可能写读比只有 1:20;
-
然后借用 Inotify(Sersync)+ Rsync 方案把NFS上的数据实时复制到本地Web服务器(静态服务器);
-
Web服务器直接读取本地的目录实时复制过来的文件,如果本地没有的话,再去NFS共享上读取文件,缓解延迟带来的问题;
-
经过实测, Inotify(Sersync)+ Rsync 方案可以单进程多线程并发处理200个100K左右的文件,而不会延迟。当然也可以开多进程,拆分监控不同的二级目录
综上所述,Rsync 结合 Inotify 为 Linux 系统提供了高效实时同步方案。通过 Inotify 监控文件变化,结合 Rsync 传输数据,可实现数据实时更新。同时,Sersync 工具在监控与同步集成上更具优势。希望本文能供实用参考。
相关文章:
Linux 实时同步服务实现(Rsync 结合 Inotify)
文章目录 1. 实时同步服务介绍2. Inotify 机制介绍3. Inotify-toolRsync 实时同步实践3.1 确认远程数据传输服务部署完成3.2 检查Linux系统是否支持Inotify实时监控3.3 安装inotify-tools3.4 命令测试3.5 重要监控事件汇总3.6 使用步骤 4. Sersync 工具使用(重点&am…...
用Java写斗地主前期工作的一些小想法
目前我们并不是要实现一个游戏,而是要对斗地主游戏做准备,主要是做牌+洗牌+发牌+给发的牌进行排序。在这个过程中我希望通过集中方式来实现: 1. 使用集合+方法+字符串的运用完成以上功能 2. 使用面向对象思想,对1做改进,主要是对其排序的改进,从而理解面向对象的真正意…...
鸿蒙数据持久化之首选项
场景介绍 用户首选项为应用提供Key-Value键值型的数据处理能力,支持应用持久化轻量级数据,并对其修改和查询。当用户希望有一个全局唯一存储的地方,可以采用用户首选项来进行存储。Preferences会将该数据缓存在内存中,当用户读取…...
将bin文件烧录到STM32
将bin文件烧录到STM32 CoFlash下载生成hex文件hex2bin使用下载bin到单片机 CoFlash下载 选择需要安装的目录 在Config中可以选择目标芯片的类型 我演示的是 stm32f103c8t6 最小系统板 Adapter:烧录器类型 Max Clock:下载速度 Por:接口类型&am…...
AtCoder Beginner Contest 397(ABCDE)
目录 A - Thermometer 翻译: 思路: 实现: B - Ticket Gate Log 翻译: 思路: 实现: C - Variety Split Easy 翻译: 思路: 实现: D - Cubes 翻译:…...
jasypt-spring-boot-starter项目如何使用jasypt加密密码
import org.jasypt.encryption.pbe.StandardPBEStringEncryptor; import org.jasypt.iv.RandomIvGenerator; import org.jasypt.salt.RandomSaltGenerator;/*** 加密密码的工具** author xxx* since 2025-03-17*/ public class JasyptTest {public static void main(String[] a…...
[AI速读]混合语言IP集成:挑战与高效解决方案
在现代SoC(系统级芯片)设计中,IP(知识产权模块)复用是提升开发效率的关键。然而,当设计涉及多种硬件描述语言(如SystemVerilog、VHDL、SystemC)时,如何高效集成不同语言的IP模块成为一大难题。本文将从实际设计场景出发,探讨混合语言IP集成的核心挑战,并介绍一套方法…...
SpringSecurity——如何获取当前登录用户的信息
目录 1. 直接注入 Principal 2. 直接注入 Authentication 3. 注入 UsernamePasswordAuthenticationToken 4. 通过 SecurityContextHolder 获取 5. 使用自定义工具方法 总结 如何获取更多的用户信息 自定义用户实体类 如何忽略某些字段(不返回前端ÿ…...
ospf动态路由
一、为什么使用动态路由 OSPF(open shortest path first开放最短路径优先)是内部网关协议(IGP)的一种,基于链路状态算法(LS)。 OSPF企业级路由协议(RFC2328 OSPFv2),核心重点协议 OSPF共三个版本,OSPFV1主要是实验室…...
在react当中利用IntersectionObserve实现下拉加载数据
目录 一、传统的下拉加载方案 二、存在问题 1.性能较差 2.不够精确 三、IntersectionObserve版本下拉加载 1、callback 2、options 四、IntersectionObserver实例 1、Intersection的优势 2、实现思路 3、代码实现 在进行前端开发的过程中,常常会碰到下拉…...
SpringBoot之如何集成SpringDoc最详细文档
文章目录 一、概念解释1、OpenAPI2、Swagger3、Springfox4、Springdoc5. 关系与区别 二、SpringDoc基本使用1、导包2、正常编写代码,不需要任何注解3、运行后访问下面的链接即可 三、SpringDoc进阶使用1、配置文档信息2、配置文档分组3、springdoc的配置参数**1. 基…...
Transaction rolled back because it has been marked as rollback-only问题解决
transaction rolled back because it has been marked as rollback-only 简略总结> 发生场景:try-catch多业务场景 发生原因:业务嵌套,事务管理混乱,外层业务与内层业务抛出异常节点与回滚节点不一致。 解决方式:修…...
单片机写的小液晶屏驱动+汉字滚屏
单片机写的小液晶屏驱动汉字滚屏 stm32f401freertos内置HZK16 单片机汉字滚屏...
SpringBoot整合LangChain4j操作AI大模型实战详解
一、引言 在当今这个人工智能飞速发展的时代,AI大模型已经逐渐渗透到我们生活的方方面面,为企业和开发者带来了前所未有的机遇。然而,如何高效地接入并利用这些AI大模型,成为了摆在许多开发者面前的一道难题。SpringBoot作为一款…...
深度解析 | Android 13 Launcher3分页指示器改造:横线变圆点实战指南
一、需求背景与技术挑战 在Android 13系统定制开发中,我们面临将Launcher3桌面从传统双层架构优化为现代单层布局的挑战。原生系统采用的分页横线指示器在视觉呈现上存在两点不足: 风格陈旧不符合Material You设计规范 空间占用较大影响屏幕利用率 通…...
【Qt】private槽函数可以被其他类中的信号连接
private槽函数可以被其他类中的信号连接。 即使 B 类的槽函数是 private 的,A 类通过信号连接 B 类的槽函数也没有问题。这是因为 Qt 的信号和槽机制是通过元对象系统(Meta-Object System)实现的,而不是直接调用函数。只要信号和…...
C语言每日一练——day_7
引言 针对初学者,每日练习几个题,快速上手C语言。第七天。(连续更新中) 采用在线OJ的形式 什么是在线OJ? 在线判题系统(英语:Online Judge,缩写OJ)是一种在编程竞赛中用…...
Python --**kwargs
在 Python 中,**kwargs 是一个特殊语法,用于在函数定义中接收任意数量的关键字参数(即键值对参数),并将这些参数以字典形式存储。它是 Python 中处理动态参数的强大工具,适用于需要灵活传递参数的场景。 1.…...
网络编程之客户端通过服务器与另外一个客户端交流
服务器使用select模型搭建,客户端1使用线程搭建,客户端2使用poll模型搭建, 使用时需要先运行服务器,具体编译可看我最后的图片 head.h头文件 #ifndef __HEAD_H_ #define __HEAD_H_ #include <stdio.h> #include <string…...
mybatis集合映射association与collection
官方文档:MyBatis的一对多关联关系 一、用途 一对一:association 一对多:collection 二、association 比较容易理解,可参考官方文档 三、collection <?xml version"1.0" encoding"UTF-8"?> &l…...
windows克隆项目找不到,修改git bash中存储的账号密码
git clone项目找不到,提示:remote: The project you were looking for could not be found. 有可能是地址错误、没有权限、账号密码错误 1.地址错误 从新检查git地址,确保地址是正确的。 2.没有权限 确保你有访问该项目的权限 3.账号密码…...
Linux中安装redis
Redis的安装包,从官方下载下来的是c语言的源码包,我们需要自己编译安装。具体操作步骤如下: 安装redis 上传redis资源包 安装C语言的编译环境 gcc yum install -y gcc-c 解压redis源码在当前目录 tar -zxvf redis-6.2.4.tar.gz 进入解压目录…...
Springboot项目打包成war包
1、首先创建一个springboot工程,然后我们改造启动类如: import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.builder.SpringApplicationBuil…...
Word:双栏排版操作步骤及注意事项
将word单栏文字排版为双栏是论文投稿前的重要步骤,也是非常容易出错的一步。但事实上排版的操作并不复杂,本篇博客旨在把操作过程及注意事项简单记录下来,便于日后浏览和查询。 【Step1】打开“显示/隐藏段落标记” (ctrl*&#…...
ArcGIS10. 8简介与安装,附下载地址
目录 ArcGIS10.8 1. 概述 2. 组成与功能 3. 10.8 特性 下载链接 安装步骤 1. 安装准备 2. 具体步骤 3.补丁 其他版本安装 ArcGIS10.8 1. 概述 ArcGIS 10.8 是由美国 Esri 公司精心研发的一款功能强大的地理信息系统(GIS)平台。其核心功能在于…...
MySQL:数据库基础
数据库基础 1.什么是数据库?2.为什么要学习数据库?3.主流的数据库(了解)4.服务器,数据库,表之间的关系5.数据的逻辑存储6.MYSQL架构7.存储引擎 1.什么是数据库? 数据库(Database,简称DB)&#x…...
数据结构-----初始数据结构、及GDB调试
一、数据结构核心概念 相互之间存在一种或多种特定关系的数据元素的集合。 1. 数据结构定义 // 嵌入式场景示例:传感器网络节点结构 struct SensorNode {uint16_t node_id; // 2字节float temperature; // 4字节uint32_t timestamp; // 4字节struct Se…...
go面向对象编程三大特性,封装、继承和多态
1.简介 go具有面向对象编程的封装、继承和多态的特性,只是实现的方式和其它OOP语言不一样,下面看下go的三大特性是如何实现的。 2.封装 2.1基本介绍 封装就是把抽象出的字段和对字段的操作封装在一起,数据被保护在内部,程序的其它包只能通过被授权的操作(方法),才能…...
华为ISC+战略规划项目数字化转型驱动的智慧供应链革新(169页PPT)(文末有下载方式)
资料解读:华为ISC战略规划项目数字化转型驱动的智慧供应链革新 详细资料请看本解读文章的最后内容。 华为的ISC战略规划项目是其供应链数字化转型的核心,旨在通过智慧供应链的革新,提升企业的竞争力和运营效率。本文将从多个维度详细解读这…...
算法刷题记录——LeetCode篇(10) [第901~1000题](持续更新)
(优先整理热门100及面试150,不定期持续更新,欢迎关注) 994. 腐烂的橘子 在给定的 m x n 网格 grid 中,每个单元格可以有以下三个值之一: 值 0 代表空单元格;值 1 代表新鲜橘子;值 2 代表腐烂的橘子。 每…...
Python中的字典:深度解析与应用实践
一、字典的本质与特性 Python字典(Dictionary)是以**键值对(Key-Value Pair)**形式存储数据的无序集合,使用大括号{}定义。其核心特性包括: 快速查找:基于哈希表实现,通过键&#…...
jmeter配件元素
jmeter配件元素 CSV Data Set Config名词解释测试场景Recycle on EOF:False配置测试结果 Recycle on EOF:True配置测试结果 Sharing mode:All Threads配置测试结果 Sharing mode:Current thread group配置测试结果 Sharing mode:Current thread配置测试结果 HTTP Header Manage…...
VSCode + CMake
参考文献: 如何用 GCC, CMake 和 Make 编译C/C代码Windows 上的 Linux 子系统:WSLWSL:桌面 UI 远程连接 RDP 配置 VScode 文章目录 CMake 配置VSCode 配置launch.jsontask.jsonc_cpp_properties.json CMake 配置 编写如下的 CmakeLists.t…...
JVM的一些知识
JVM简介 JVM 是 Java Virtual Machine 的简称,意为 Java 虚拟机。 虚拟机是指通过软件模拟的具有完整硬件功能的、运行在一个完全隔离的环境中的完整计算机系统。常见的虚拟机:JVM、VMwave、Virtual Box。 JVM 和其他两个虚拟机的区别: VMw…...
SpringCloud网关:Gateway路由配置与过滤器链
文章目录 引言一、Gateway基本架构二、路由配置方式2.1 配置文件方式2.2 Java代码方式 三、内置断言工厂四、内置过滤器工厂4.1 请求路径相关过滤器4.2 请求和响应头过滤器4.3 功能性过滤器 五、自定义过滤器5.1 自定义GatewayFilter5.2 自定义过滤器工厂 六、全局过滤器总结 引…...
Git 分支使用规范全解(多人协作开发适用)
🚀 Git 分支使用规范全解(多人协作开发适用) 本文将为你梳理一套清晰、标准、适合企业/团队使用的 Git 分支管理策略,适用于前后端、边缘端、AI项目等多种场景。 🧩 为什么要规范分支管理? 防止多人协作混乱、冲突频发清晰区分:开发中 / 待发布 / 已上线 的版本快速定…...
【华三】路由器交换机忘记登入密码或super密码的重启操作
【华三】路由器交换机忘记登入密码或super密码的重启操作 背景步骤跳过认证设备:路由器重启设备翻译说明具体操作 跳过当前系统配置重启设备具体操作 背景 当console口的密码忘记,或者说本地用户的密码忘记,其实这时候是登入不了路由器的&am…...
Linux驱动开发进阶 - 文件系统
文章目录 1、前言2、学习目标3、VFS虚拟文件系统3.1、超级块(Super Block)3.2、dentry3.3、inode3.4、file 4、文件系统的挂载5、文件系统的注册5.1、文件系统的注册过程5.1.2、定义文件系统类型5.1.3、注册文件系统5.1.4、注销文件系统 5.2、文件系统的…...
Mac:JMeter 下载+安装+环境配置(图文详细讲解)
📌 下载JMeter 下载地址:https://jmeter.apache.org/download_jmeter.cgi 📌 无需安装 Apache官网下载 JMeter 压缩包,无需安装,下载解压后放到自己指定目录下即可。 按我自己的习惯,我会在用户 jane 目…...
蓝桥杯备考:图论之Prim算法
嗯。通过我们前面的学习,我们知道了,一个具有n个顶点的连通图,它的生成树包括n-1个边,如果边多一条就会变成图,少一条就不连通了 接下来我们来学一下把图变成生成树的一个算法 Prim算法,我们从任意一个结…...
Linux 文件操作-标准IO函数3- fread读取、fwrite写入、 fprintf向文件写入格式化数据、fscanf逐行读取格式化数据的验证
目录 1. fread 从文件中读取数据 1.1 读取次数 每次读取字节数 < 原内容字节数 1.2 读取次数 每次读取字节数 > 原内容字节数 2.fwrite 向文件中写入数据 2.1写入字符串验证 2.2写入结构体验证 3. fprintf 将数据写入到指定文件 4. fscanf 从文件中逐行读取内容…...
汽车一键启动系统使用方便,舒适出行,轻松匹配
汽车一键启动系统 系统定义 移动管家汽车一键启动系统是装置在智能汽车上的一部分,是实现简约打火和熄火过程的一个按钮装置。它可以在原车钥匙锁头的位置改装,也能独立面板改装,现在很多高低配置的车辆都可安装。 功能特点 基本功能 启…...
python函数的多种参数使用形式
目录 1. 位置参数(Positional Arguments) 2. 关键字参数(Keyword Arguments) 3. 默认参数(Default Arguments) 4. 可变参数(Variable Positional Arguments) 5. 关键字可变参数&…...
Qt带参数的信号和槽,以及信号与槽的连接方式
1.带参数的信号和槽 Qt的信号与槽也支持带有参数,同时也可以支持重载 此处我们要求,信号函数的参数列表要和对应连接的槽函数参数列表一致 此时信号触发,调用到槽函数的时候,信号函数中的实参就能够被传递到槽函数的形参中 示例…...
深度解析ECharts.js:构建现代化数据可视化的利器
引言:数据可视化的新时代挑战 在数字化转型浪潮中,数据可视化已成为企业决策和用户体验的关键环节。面对海量数据的呈现需求,传统表格已无法满足用户对直观洞察的渴求。作为百度开源的JavaScript可视化库,ECharts.js凭借其强大的…...
Flutter:页面滚动,导航栏背景颜色过渡动画
记录:导航默认透明,页面发生滚动后,导航背景色由0-1,过渡到白色背景。 view import package:ducafe_ui_core/ducafe_ui_core.dart; import package:flutter/material.dart; import package:get/get.dart; import package:redo…...
一文了解ThreadLocal
什么是ThreadLocal? ThreadLocal是每个线程私有的,线程可以把自己的私有数据放到ThreadLocal里面,不用担心其他线程访问到自己ThreadLocal。 通过set()方法将值存入ThreadLocal或者修改值,get()方法取出值,remove()方…...
日常学习开发记录-input组件
实现 1.实现2.inline-table和table-cell实现2.1 表格布局的特性2.2 示例 3.clear清除事件未生效3.1 原因3.2 解决 4. 增加type为text和textarea4.1 rows,autosize的实现 5.拓展-composition事件 1.实现 <template><div class"my-input":class"{is-dis…...
【数据库系统原理】简答题
真题 2024-10 31.数据模型的三大要素是什么? 32.简述关系模型的参照完整性规则。 33.什么是视图?视图的作用是什么? 34.简述两个实体型之间联系的三种形式,并举例说明。 35.什么是数据库备份?MySQL使用什么语句实现备份与恢复数据库中表的数据? 2024-04 31.请解释数据…...
20250319在荣品的PRO-RK3566开发板的buildroot系统下使用集成的QT应用调试串口UART3
stty -F /dev/ttyS3 115200 -echo cat /dev/ttyS3 & echo serialdata > /dev/ttyS3 20250319在荣品的PRO-RK3566开发板的buildroot系统下使用集成的QT应用调试串口UART3 2025/3/19 14:17 缘起:在荣品的PRO-RK3566开发板的buildroot系统下,在命令…...