当前位置: 首页 > news >正文

Linux部署Redis Cluster高可用集群(附带集群节点添加删除以及槽位分配操作详解)

目录

    • 一、前言
    • 二、下载安装Redis
      • 2.1、选择需要安装的Redis版本
      • 2.2、下载并解压Redis
      • 2.3、编译安装Redis
    • 三、部署Redis Cluster高可用集群
      • 3.1、准备配置文件
      • 3.2、启动Redis服务
      • 3.3、创建Redis集群
      • 3.4、查看集群关系
      • 3.5、连接集群Redis进行数据读写以及重定向测试
      • 3.6、故障转移和恢复测试
      • 3.7、Redis集群中添加节点并分配槽位
        • 3.7.1、先部署两个新的Redis服务使用端口号7007 7008
        • 3.7.2、集群中添加主节点
        • 3.7.3、给新添加的主节点分配槽位
        • 3.7.4、集群中给指定主节点添加从节点
      • 3.8、删除Redis集群中节点
        • 3.8.1、删除从节点
        • 3.8.2、将被删除主节点槽位迁移到别的主节点
        • 3.8.3、删除主节点
    • 四、SpringBoot集成Redis Cluster集群

一、前言

Redis Cluster 集群是一个由多个主从节点群组成的分布式服务器群,它具有复制、高可用和分片特性。Redis集群不需要sentinel哨兵也能完成节点移除和故障转移的功能。需要将每个节点设置成集群模式,这种集群模式没有中心节点,可水平扩展,需要将每个节点设置成集群模式,这种集群模式没有中心节点,可水平扩展,据官方文档称可以线性扩展到上万个节点(官方推荐不超过1000个节点)。redis集群的性能和高可用性均优于之前版本的哨兵模式,且集群配置比较简单,Redis Cluster 将所有数据划分为 16384个槽位,每个redis 节点负责其中一部分槽位,客户端为了可以直接定位(对 key 通过 crc16 进行 hash 再对16384取余)某个具体的 key 所在节点,需要缓存槽位相关信息,这样才可以准确快速地定位到相应的节点。同时因为可能会存在客户端与服务器存储槽位的信息不一致的情况,还需要纠正机制(通过返回 -MOVED 3999 127.0.0.1:6379,客户端收到后需要立即纠正本地的槽位映射表)来实现槽位信息的校验调整.在这里插入图片描述

这里重点做Redis Cluster集群部署,要想了解原理可以跳转,Redis Cluster高可用集群原理。

二、下载安装Redis

2.1、选择需要安装的Redis版本

在官网发行版中选择一个自己需要的版本,我这里使用Redis6.2.0,小版本区别不大。
Redis发行版本列表:https://download.redis.io/releases
在这里插入图片描述

2.2、下载并解压Redis

  • 下载redis到/usr/local/redis目录中,这里使用wget下载,如果没有wget执行yum -y install wget安装
wget -P /usr/local/redis http://download.redis.io/releases/redis-6.2.0.tar.gz
  • 进入/usr/local/redis目录解压redis
cd /usr/local/redis
tar -xzf redis-6.2.0.tar.gz

在这里插入图片描述

2.3、编译安装Redis

  • 安装gcc
# redis是c语言编写的,编译Redis需要gcc环境
yum -y install gcc
# 安装好后查看gcc信息
gcc -v

在这里插入图片描述

  • 编译
# 进入redis-6.2.0目录
cd redis-6.2.0
# 执行make开始编译 因为我们下载的是源码需要编译成可执行文件
make

在这里插入图片描述

  • 安装
# 编译成功后,执行make PREFIX=/usr/local/redis/redis-6.2.0 install 安装redis
# 这里指定目录之后,方便后续卸载,直接rm -rf /usr/local/redis/redis-6.2.0 即可删除redis
make PREFIX=/usr/local/redis/redis-6.2.0 install
PREFIX 这个关键字的作用是安装的时候用于指定程序存放的路径,假设不添加该关键字Linux会将:可执行文件存放在/usr/local/bin目录;库文件会存放在/usr/local/lib目录;配置文件会存放在/usr/local/etc目录;其他的资源文件会存放在usr/local/share目录

在这里插入图片描述

三、部署Redis Cluster高可用集群

  • 这里使用单台主机部署,部署6个redis实例,3个主节点,每个主节点1个从节点,端口号使用7001-7006,Redis服务端口和集群节点gossip通信端口默认是在Redis端口号上加10000,如果不是同一台主机要将防火墙关闭或者开放对应端口。
  • 在Redis Cluster集群部署中不用自己指定master节点,在创建集群时会根据Redis服务数量还有指定每个master节点需要有多少个slave节点自动分配,在下面会进行演示。

3.1、准备配置文件

  • 1、创建每个节点信息的存储目录
# 创建集群总的存储目录
mkdir /usr/local/redis/redis-6.2.0/cluster
# 进入总存储目录
cd /usr/local/redis/redis-6.2.0/cluster
# 创建每个节点存储目录
mkdir 7001 7002 7003 7004 7005 7006
  • 2、先准备7001节点的配置文件,后面只要替换配置文件中的7001然后分发到不同节点存储目录即可
# 进入 usr/local/redis/redis-6.2.0 目录操作
cd /usr/local/redis/redis-6.2.0
# 备份一份当前目录redis.conf文件做留存
cp redis.conf redis.conf.back
# 修改redis.conf文件 先给7001节点做配置,后面只要替换配置文件中的7001即可
vi redis.conf

修改下面配置

# 绑定的主机地址,建议填redis安装服务器的业务私网地址
# 这里为了方便直接找到这个配置注释了
#bind 127.0.0.1 -::1# redis进程的端口号 默认是6379
port 7001# 是否开启保护模式,默认开启。要是配置里没有指定bind和密码,开启该参数后,redis只会本地进行访问,拒绝外部访问,这里关闭保护模式设置为no。
protected-mode no# redis进程是否以守护进程的方式运行,yes为是(后台运行),no为否(不以守护进程的方式运行会占用一个终端,终端关闭redis服务也会关闭)。
daemonize yes# 指定redis进程的PID文件存放位置
pidfile /var/run/redis_7001.pid# log文件输出位置
logfile /usr/local/redis/redis-6.2.0/cluster/7001/redis.log# 指定数据文件存放目录(默认当前启动时所在目录),因为我这里是一台机器部署的要区分开每个节点持久化文件存放的目录
dir /usr/local/redis/redis-6.2.0/cluster/7001# 设置redis连接密码,如果配置了连接密码,客户端在连接redis是需要通过AUTH<password>命令提供密码,默认关闭
requirepass 123456# 设置集群节点间访问密码,跟上面一致
masterauth 123456# 启动集群模式
cluster-enabled yes# 集群节点信息文件,这里700x最好和port对应上
cluster-config-file nodes‐7001.conf# 集群节点的超时时限,节点发送对其它节点的PING命令的时候如果超过这个时间还没有响应则标记另外一个为故障状态 默认:15000毫秒
cluster‐node‐timeout 15000
  • 3、分发各节点配置文件并替换对应端口
# 复制7001的redis.conf到每个节点数据存储目录
cp redis.conf ./cluster/7001/redis.conf
cp redis.conf ./cluster/7002/redis.conf
cp redis.conf ./cluster/7003/redis.conf
cp redis.conf ./cluster/7004/redis.conf
cp redis.conf ./cluster/7005/redis.conf
cp redis.conf ./cluster/7006/redis.conf# 替换每个节点配置文件中的7001字符串为自己节点端口
sed -i  "s/7001/7002/g" ./cluster/7002/redis.conf
sed -i  "s/7001/7003/g" ./cluster/7003/redis.conf
sed -i  "s/7001/7004/g" ./cluster/7004/redis.conf
sed -i  "s/7001/7005/g" ./cluster/7005/redis.conf
sed -i  "s/7001/7006/g" ./cluster/7006/redis.conf

3.2、启动Redis服务

./bin/redis-server ./cluster/7001/redis.conf
./bin/redis-server ./cluster/7002/redis.conf
./bin/redis-server ./cluster/7003/redis.conf
./bin/redis-server ./cluster/7004/redis.conf
./bin/redis-server ./cluster/7005/redis.conf
./bin/redis-server ./cluster/7006/redis.conf

查看进程启动状态

ps -aux | grep redis

在这里插入图片描述

3.3、创建Redis集群

Redis5以前的版本集群是依靠ruby脚本redis‐trib.rb实现,之后是使用redis‐cli创建整个Redis集群,我这里使用的是Redis6.2.0.

# 创建集群
## -a 密码
## --cluster create 创建集群后面带每个节点的IP:端口
## --cluster-replicas 1 表示为每个master创建一个slave节点
./bin/redis-cli -a 123456 --cluster create 172.16.8.186:7001 172.16.8.186:7002 172.16.8.186:7003 172.16.8.186:7004 172.16.8.186:7005 172.16.8.186:7006 --cluster-replicas 1

在创建时会询问是否设置成列出来的配置,输入yes回车即可,槽位分配和主从关系都是由Cluster计算出来的,后续可以通过命令调整

[root@parent-docker redis-6.2.0]# ./bin/redis-cli -a 123456 --cluster create 172.16.8.186:7001 172.16.8.186:7002 172.16.8.186:7003 172.16.8.186:7004 172.16.8.186:7005 172.16.8.186:7006 --cluster-replicas 1
Warning: Using a password with '-a' or '-u' option on the command line interface may not be safe.
>>> Performing hash slots allocation on 6 nodes...
Master[0] -> Slots 0 - 5460
Master[1] -> Slots 5461 - 10922
Master[2] -> Slots 10923 - 16383
Adding replica 172.16.8.186:7005 to 172.16.8.186:7001
Adding replica 172.16.8.186:7006 to 172.16.8.186:7002
Adding replica 172.16.8.186:7004 to 172.16.8.186:7003
>>> Trying to optimize slaves allocation for anti-affinity
[WARNING] Some slaves are in the same host as their master
M: e4528f6cbf114daaa6d18c8a4111a98329b7c319 172.16.8.186:7001slots:[0-5460] (5461 slots) master
M: d52909707e7a7946955759e9b943e81fe85b3d67 172.16.8.186:7002slots:[5461-10922] (5462 slots) master
M: aa60b733ba646f9bc541f0a29f9ccaffe9e65f09 172.16.8.186:7003slots:[10923-16383] (5461 slots) master
S: d524a41b8a358b7ad678ff913da62899668d2e69 172.16.8.186:7004replicates e4528f6cbf114daaa6d18c8a4111a98329b7c319
S: 5fb34e0d2187e49cfbba3b4fb89bf40d591089a0 172.16.8.186:7005replicates d52909707e7a7946955759e9b943e81fe85b3d67
S: d1eea9da1d8f471aa90e12f5957f538bce72312c 172.16.8.186:7006replicates aa60b733ba646f9bc541f0a29f9ccaffe9e65f09
Can I set the above configuration? (type 'yes' to accept): yes
>>> Nodes configuration updated
>>> Assign a different config epoch to each node
>>> Sending CLUSTER MEET messages to join the cluster
Waiting for the cluster to join
.
>>> Performing Cluster Check (using node 172.16.8.186:7001)
M: e4528f6cbf114daaa6d18c8a4111a98329b7c319 172.16.8.186:7001slots:[0-5460] (5461 slots) master1 additional replica(s)
M: d52909707e7a7946955759e9b943e81fe85b3d67 172.16.8.186:7002slots:[5461-10922] (5462 slots) master1 additional replica(s)
S: d524a41b8a358b7ad678ff913da62899668d2e69 172.16.8.186:7004slots: (0 slots) slavereplicates e4528f6cbf114daaa6d18c8a4111a98329b7c319
S: 5fb34e0d2187e49cfbba3b4fb89bf40d591089a0 172.16.8.186:7005slots: (0 slots) slavereplicates d52909707e7a7946955759e9b943e81fe85b3d67
M: aa60b733ba646f9bc541f0a29f9ccaffe9e65f09 172.16.8.186:7003slots:[10923-16383] (5461 slots) master1 additional replica(s)
S: d1eea9da1d8f471aa90e12f5957f538bce72312c 172.16.8.186:7006slots: (0 slots) slavereplicates aa60b733ba646f9bc541f0a29f9ccaffe9e65f09
[OK] All nodes agree about slots configuration.
>>> Check for open slots...
>>> Check slots coverage...
[OK] All 16384 slots covered.

3.4、查看集群关系

在任意一台节点都能查看

./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7001

可以在这个信息中看到主从关系和槽位以及每个主节点中key的数量信息。
在这里插入图片描述

还可以连接上任意一台Redis服务使用命令查看

# 查看集群信息
172.16.8.186:7001> cluster info
# 查看节点列表
172.16.8.186:7001> cluster nodes

3.5、连接集群Redis进行数据读写以及重定向测试

  • 1、连接集群Redis(连接任意节点都可以,操作key时如果槽位不在当前节点或者不是主节点,会自动重定向到对应的主节点)
## -c 集群节点跳转重定向,如果不加-c,节点之间是无法自动跳转的
./bin/redis-cli -c -h 172.16.8.186 -p 7001 -a 123456
  • 2、插入数据
172.16.8.186:7001> set names kerwin

在7001节点执行插入操作,计算出names这个key的槽位为6659,是在主节点7002中,这里可以看到直接重定向到了7002节点,就相当于在7002节点执行的插入,而且客户端这里连接的节点也变为了7002,如果计算出的槽位在7001中则可以直接插入。
在这里插入图片描述

  • 3、读取数据
    先关闭连接,然后重新连接。

    172.16.8.186:7001> get names
    

    读数据这里和写数据是一样的逻辑,计算出key的槽位不在当前主节点,则会自动重定向到对应的主节点中读取。
    在这里插入图片描述

  • 4、从节点读数据
    根据上面的测试我们看到names这个key会落在7002节点中,7002的从节点是7005,那么我们在7005上执行读操作会进行重定向吗,这里测试一下。

    ## -c 集群节点跳转重定向,如果不加-c,节点之间是无法自动跳转的
    ./bin/redis-cli -c -h 172.16.8.186 -p 7005 -a 123456
    

    这里可以看到就算在从节点也无法获取数据,还是会重定向到主节点获取数据,官方默认设置的是不分担读请求的、只作备份和故障转移用,当有请求读向从节点时,会被重定向对应的主节点来处理。
    在这里插入图片描述

    要想在从节点读取数据也可以使用readonly命令,这个readonly告诉 Redis Cluster 从节点客户端愿意读取可能过时的数据并且对写请求不感兴趣,只在当前会话有效,断开连接后readonly就失效了,再次连接需要重新使用该命令。
    在这里插入图片描述

3.6、故障转移和恢复测试

  • 1、这里我们手动停止主节点7001
./bin/redis-cli -c -h 172.16.8.186 -p 7001 -a 123456 shutdown
  • 2、查看集群信息
./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7002

这里可以看到7001已经没了,7004变成了新的主节点
在这里插入图片描述

  • 3、重新启动7001
./bin/redis-server ./cluster/7001/redis.conf
  • 4、查看集群信息
./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7002

可以看到7001又加入了集群,不过这个时候是7004的从节点
在这里插入图片描述

3.7、Redis集群中添加节点并分配槽位

进入/usr/local/redis/redis-6.2.0目录操作cd /usr/local/redis/redis-6.2.0

3.7.1、先部署两个新的Redis服务使用端口号7007 7008
# 1、创建7007 7008节点存储目录
mkdir /usr/local/redis/redis-6.2.0/cluster/7007
mkdir /usr/local/redis/redis-6.2.0/cluster/7008# 2、复制7001的redis.conf到7007 7008节点存储目录
cp redis.conf ./cluster/7007/redis.conf
cp redis.conf ./cluster/7008/redis.conf# 3、替换每个节点配置文件中的7001字符串为自己节点端口
sed -i  "s/7001/7007/g" ./cluster/7007/redis.conf
sed -i  "s/7001/7008/g" ./cluster/7008/redis.conf# 4、启动7007 7008节点Redis服务
./bin/redis-server ./cluster/7007/redis.conf
./bin/redis-server ./cluster/7008/redis.conf# 5、查看7007 7008节点启动状态
ps -aux | grep redis

在这里插入图片描述

3.7.2、集群中添加主节点
# --cluster add-node  添加节点,如果命令中没有--cluster-slave --cluster-master-id则是添加主节点
# 172.16.8.186:7007  需要添加的Redis服务
# 172.16.8.186:7001  集群中任意一个节点都可以,我这里使用7001节点
# -a 123456  Redis服务密码
./bin/redis-cli -a 123456 --cluster add-node 172.16.8.186:7007 172.16.8.186:7001

在这里插入图片描述

添加好后可以查看集群信息

./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7001

可以看到7007已经成功加入集群,并且为主节点,但是没有槽位,无法进行读写,需要先分配槽位
在这里插入图片描述

3.7.3、给新添加的主节点分配槽位
# --cluster reshard  重新分配槽位
# 172.16.8.186:7001  集群中任意节点都行,这里选择7001
./bin/redis-cli -a 123456 --cluster reshard 172.16.8.186:7001

在这里插入图片描述

回车执行后还会询问一下是否执行分配,输入yes回车即可,后面会开始执行槽位迁移,执行完后查看集群信息,可以看到7007节点被分配了三个区间的槽位,总合4096,正好是从其它3个节点迁移过来的。
在这里插入图片描述

3.7.4、集群中给指定主节点添加从节点
# --cluster add-node  添加节点,如果命令中加了--cluster-slave --cluster-master-id 则是添加从节点
# --cluster-slave 添加从节点
# --cluster-master-id 主节点ID
# 172.16.8.186:7008  需要添加的Redis服务
# 172.16.8.186:7001  集群中任意一个节点都可以,我这里使用7001节点
# -a 123456  Redis服务密码
./bin/redis-cli -a 123456 --cluster add-node 172.16.8.186:7008 172.16.8.186:7001 --cluster-slave --cluster-master-id 5292ce107ed07159d3e9b413aad3210277494c9c

在这里插入图片描述

添加好查看一下集群节点信息

./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7001

从节点也添加成功
在这里插入图片描述

3.8、删除Redis集群中节点

  • 注意事项
    • 要删除节点首先要先删除从节点,确保主节点下没有从节点,避免从节点自动切换成其它主节点的从节点。

    • 删除主节点前要先将被删除节点的槽位分配出去,不然无法删除会出现[ERR] Node 172.16.8.186:7007 is not empty! Reshard data away and try again.

    • 在移除某个Redis节点之前,不能登入该节点操作,要使用其它节点会话操作,否则不能正常移除该节点。

3.8.1、删除从节点
# --cluster del-node 删除节点
# 172.16.8.186:7001 集群中任意节点,不能是被删除的节点
# 7d1437adb68606d46160220cdba2f2728b88ec4d 在最后面带上要删除的节点ID
./bin/redis-cli -a 123456 --cluster del-node 172.16.8.186:7001 7d1437adb68606d46160220cdba2f2728b88ec4d
3.8.2、将被删除主节点槽位迁移到别的主节点
# --cluster reshard  重新分配槽位
# 172.16.8.186:7001  集群中任意节点都行,这里选择7001
./bin/redis-cli -a 123456 --cluster reshard 172.16.8.186:7001

在这里插入图片描述

查看一下集群节点信息

./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7001

在这里插入图片描述

3.8.3、删除主节点
# --cluster del-node 删除节点
# 172.16.8.186:7001 集群中任意节点,不能是被删除的节点
# 5292ce107ed07159d3e9b413aad3210277494c9c 在最后面带上要删除的节点ID
./bin/redis-cli -a 123456 --cluster del-node 172.16.8.186:7001 5292ce107ed07159d3e9b413aad3210277494c9c

在这里插入图片描述

查看一下集群节点信息

./bin/redis-cli -a 123456 --cluster check 172.16.8.186:7001

7007和7008节点都被移除
在这里插入图片描述

四、SpringBoot集成Redis Cluster集群

https://blog.csdn.net/weixin_44606481/article/details/134073096

相关文章:

Linux部署Redis Cluster高可用集群(附带集群节点添加删除以及槽位分配操作详解)

目录 一、前言二、下载安装Redis2.1、选择需要安装的Redis版本2.2、下载并解压Redis2.3、编译安装Redis 三、部署Redis Cluster高可用集群3.1、准备配置文件3.2、启动Redis服务3.3、创建Redis集群3.4、查看集群关系3.5、连接集群Redis进行数据读写以及重定向测试3.6、故障转移和…...

【PWN · heap | Off-By-One】Asis CTF 2016 b00ks

萌新进度太慢了&#xff0c;才真正开始heap&#xff0c;还是从简单的Off-By-One开始吧 前言 步入堆的学习。堆的知识复杂而多&#xff0c;于是想着由wiki从简单部分逐个啃。 b00ks是经典的堆上off-by-one漏洞题目。刚开始看很懵&#xff08;因为确实连堆的管理机制都没有完全…...

C++STL---Vector、List所要掌握的基本知识

绪论​ 拼着一切代价&#xff0c;奔你的前程。 ——巴尔扎克&#xff1b;本章主要围绕vector和list的使用&#xff0c;以及容器底层迭代器失效问题&#xff0c;同时会有对原码的分析和模拟实现其底层类函数。​​​​话不多说安全带系好&#xff0c;发车啦&#xff08;建议电脑…...

使用FastAPI部署Ultralytics YOLOv5模型

YOLO是You Only Look Once(你只看一次)的缩写&#xff0c;它具有识别图像中的物体的非凡能力&#xff0c;在日常应用中会经常被使用。所以在本文中&#xff0c;我们将介绍如何使用FastAPI的集成YOLOv5&#xff0c;这样我们可以将YOLOv5做为API对外提供服务。 Python有几个web框…...

A. Doremy‘s Paint 3

今天第一次打CF&#xff0c;不过鼠鼠被气死了 先说说战况&#xff0c;今天一发没A&#xff08;赛场上&#xff09;&#xff0c;生活真是无奈&#xff0c;废物女友真是一点用没有 心里也很烦&#xff0c;什么压力都自己扛着。每天想尝试改变什么&#xff0c;又被现实掣肘&…...

深度学习_1 介绍;安装环境

深度学习 学习自李沐老师的课程。笔记主要以总结老师所讲解的内容以及我个人的想法为主&#xff0c;侵删&#xff01; 课程链接&#xff1a;课程安排 - 动手学深度学习课程 (d2l.ai) 介绍 AI地图&#xff1a; 首先&#xff0c;AI 能对问题处理到什么地步&#xff1f;分为四…...

Python基础入门例程19-NP19 列表的长度(列表)

最近的博文&#xff1a; Python基础入门例程18-NP18 生成数字列表&#xff08;列表&#xff09;-CSDN博客 Python基础入门例程17-NP17 生成列表(列表)-CSDN博客 Python基础入门例程16-NP16 发送offer(列表)-CSDN博客 目录 描述 输入描述&#xff1a; 输出描述&#xff1…...

LeetCode 2558. 从数量最多的堆取走礼物

【LetMeFly】2558.从数量最多的堆取走礼物 力扣题目链接&#xff1a;https://leetcode.cn/problems/take-gifts-from-the-richest-pile/ 给你一个整数数组 gifts &#xff0c;表示各堆礼物的数量。每一秒&#xff0c;你需要执行以下操作&#xff1a; 选择礼物数量最多的那一…...

【JVM】字节码文件的组成部分

&#x1f40c;个人主页&#xff1a; &#x1f40c; 叶落闲庭 &#x1f4a8;我的专栏&#xff1a;&#x1f4a8; c语言 数据结构 javaEE 操作系统 Redis 石可破也&#xff0c;而不可夺坚&#xff1b;丹可磨也&#xff0c;而不可夺赤。 JVM 一、字节码文件的组成部分1.1 iconst_0…...

STM32 TIM(四)编码器接口

STM32 TIM&#xff08;四&#xff09;编码器接口 编码器接口简介 Encoder Interface 编码器接口 编码器接口可接收增量&#xff08;正交&#xff09;编码器的信号&#xff0c;根据编码器旋转产生的正交信号脉冲&#xff0c;自动控制CNT自增或自减&#xff0c;从而指示编码器的…...

力扣第56题 合并区间 c++ 贪心

题目 56. 合并区间 中等 相关标签 数组 排序 以数组 intervals 表示若干个区间的集合&#xff0c;其中单个区间为 intervals[i] [starti, endi] 。请你合并所有重叠的区间&#xff0c;并返回 一个不重叠的区间数组&#xff0c;该数组需恰好覆盖输入中的所有区间 。 示例…...

php 日期

其中关于周的起止&#xff0c;使用date("N")&#xff0c;确保每周周一为起始&#xff0c;避免周日时出现作为新一周起始的情况 //获取上个月第一天 echo "上个月开始时间&#xff1a;".date(Y-m-01 00:00:00,strtotime(-1 month))."\r\n\r\n"; …...

食物链解读

[NOI2001] 食物链 题目描述 动物王国中有三类动物 A , B , C A,B,C A,B,C&#xff0c;这三类动物的食物链构成了有趣的环形。 A A A 吃 B B B&#xff0c; B B B 吃 C C C&#xff0c; C C C 吃 A A A。 现有 N N N 个动物&#xff0c;以 1 ∼ N 1 \sim N 1∼N 编号。…...

Day10配置文件日志多线程

配置文件 在企业开发过程中&#xff0c;我们习惯把一些需要灵活配置的数据放在一些文本文件中&#xff0c;而不是在Java代码写死 我们把这种存放程序配置信息的文件&#xff0c;统称为配置文件 properties 是一个Map集合&#xff08;键值对集合&#xff09;&#xff0c;但是我…...

leetcode:1154. 一年中的第几天(python3解法)

难度&#xff1a;简单 给你一个字符串 date &#xff0c;按 YYYY-MM-DD 格式表示一个 现行公元纪年法 日期。返回该日期是当年的第几天。 示例 1&#xff1a; 输入&#xff1a;date "2019-01-09" 输出&#xff1a;9 解释&#xff1a;给定日期是2019年的第九天。 示例…...

竞赛 深度学习图像修复算法 - opencv python 机器视觉

文章目录 0 前言2 什么是图像内容填充修复3 原理分析3.1 第一步&#xff1a;将图像理解为一个概率分布的样本3.2 补全图像 3.3 快速生成假图像3.4 生成对抗网络(Generative Adversarial Net, GAN) 的架构3.5 使用G(z)生成伪图像 4 在Tensorflow上构建DCGANs最后 0 前言 &#…...

flutter升级+生成drift文件

1. flutter升级 可以安装fvm进行flutter version manager FVM 安装笔记 - 掘金 (juejin.cn) 使用flutter upgrade, 但是没有效果&#xff0c; 可能需要到我的电脑中&#xff0c;更改高级系统设置&#xff1b;改变/增加环境变量&#xff1b;用来加上flutter官网获取信息的内…...

[AUTOSAR][诊断管理][ECU][$34] 下载请求

文章目录 一、简介二、服务请求报文定义肯定响应支持的NRC三、示例代码34_req_dowload.c一、简介 RequestDownload(0x34)—— 下载请求 这个服务主要是用来给ECU下载数据的,最常见的应用就是在bootloader中,程序下载工具会发起下载请求,以完成ECU程序的升级。 二、服务…...

C 标准库 - <errno.h>和<float.h>详解

目录 简介 常见库宏 简介 常见库宏 <errno.h> 简介 <errno.h>头文件定义了一个名为errno的全局变量&#xff0c;用于表示最近发生的错误代码。errno是一个整数变量&#xff0c;它的值通常是一个非零的错误代码&#xff0c;用于指示发生了什么类型的错误。也可以…...

对于如何学习的一点思考

目录 1、学习遇到的问题 2、问题分析 3、解决思路 1、学习遇到的问题 我们经常在学习一个知识时&#xff0c;经常会遇到知识点凌乱、读书效率低、缺乏长期记忆等问题&#xff0c;主要体现在&#xff1a; 知识点凌乱&#xff1a;花时间学习了很多技术点&#xff0c;但是由于…...

centos 7 部署awstats 网站访问检测

一、基础环境准备&#xff08;两种安装方式都要做&#xff09; bash # 安装必要依赖 yum install -y httpd perl mod_perl perl-Time-HiRes perl-DateTime systemctl enable httpd # 设置 Apache 开机自启 systemctl start httpd # 启动 Apache二、安装 AWStats&#xff0…...

定时器任务——若依源码分析

分析util包下面的工具类schedule utils&#xff1a; ScheduleUtils 是若依中用于与 Quartz 框架交互的工具类&#xff0c;封装了定时任务的 创建、更新、暂停、删除等核心逻辑。 createScheduleJob createScheduleJob 用于将任务注册到 Quartz&#xff0c;先构建任务的 JobD…...

CocosCreator 之 JavaScript/TypeScript和Java的相互交互

引擎版本&#xff1a; 3.8.1 语言&#xff1a; JavaScript/TypeScript、C、Java 环境&#xff1a;Window 参考&#xff1a;Java原生反射机制 您好&#xff0c;我是鹤九日&#xff01; 回顾 在上篇文章中&#xff1a;CocosCreator Android项目接入UnityAds 广告SDK。 我们简单讲…...

【单片机期末】单片机系统设计

主要内容&#xff1a;系统状态机&#xff0c;系统时基&#xff0c;系统需求分析&#xff0c;系统构建&#xff0c;系统状态流图 一、题目要求 二、绘制系统状态流图 题目&#xff1a;根据上述描述绘制系统状态流图&#xff0c;注明状态转移条件及方向。 三、利用定时器产生时…...

MySQL 8.0 OCP 英文题库解析(十三)

Oracle 为庆祝 MySQL 30 周年&#xff0c;截止到 2025.07.31 之前。所有人均可以免费考取原价245美元的MySQL OCP 认证。 从今天开始&#xff0c;将英文题库免费公布出来&#xff0c;并进行解析&#xff0c;帮助大家在一个月之内轻松通过OCP认证。 本期公布试题111~120 试题1…...

Hive 存储格式深度解析:从 TextFile 到 ORC,如何选对数据存储方案?

在大数据处理领域&#xff0c;Hive 作为 Hadoop 生态中重要的数据仓库工具&#xff0c;其存储格式的选择直接影响数据存储成本、查询效率和计算资源消耗。面对 TextFile、SequenceFile、Parquet、RCFile、ORC 等多种存储格式&#xff0c;很多开发者常常陷入选择困境。本文将从底…...

并发编程 - go版

1.并发编程基础概念 进程和线程 A. 进程是程序在操作系统中的一次执行过程&#xff0c;系统进行资源分配和调度的一个独立单位。B. 线程是进程的一个执行实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位。C.一个进程可以创建和撤销多个线程;同一个进程中…...

Linux nano命令的基本使用

参考资料 GNU nanoを使いこなすnano基础 目录 一. 简介二. 文件打开2.1 普通方式打开文件2.2 只读方式打开文件 三. 文件查看3.1 打开文件时&#xff0c;显示行号3.2 翻页查看 四. 文件编辑4.1 Ctrl K 复制 和 Ctrl U 粘贴4.2 Alt/Esc U 撤回 五. 文件保存与退出5.1 Ctrl …...

海云安高敏捷信创白盒SCAP入选《中国网络安全细分领域产品名录》

近日&#xff0c;嘶吼安全产业研究院发布《中国网络安全细分领域产品名录》&#xff0c;海云安高敏捷信创白盒&#xff08;SCAP&#xff09;成功入选软件供应链安全领域产品名录。 在数字化转型加速的今天&#xff0c;网络安全已成为企业生存与发展的核心基石&#xff0c;为了解…...

数据分析六部曲?

引言 上一章我们说到了数据分析六部曲&#xff0c;何谓六部曲呢&#xff1f; 其实啊&#xff0c;数据分析没那么难&#xff0c;只要掌握了下面这六个步骤&#xff0c;也就是数据分析六部曲&#xff0c;就算你是个啥都不懂的小白&#xff0c;也能慢慢上手做数据分析啦。 第一…...