Redis6学习笔记
创始人
2024-05-30 02:35:43
0

文章目录

  • 一、简介
    • 1.1 Redis键(key)
    • 1.2 字符串(String)
      • 1.2.1 简介
      • 1.2.2 命令
      • 1.2.3 数据结构
    • 1.3 列表(List)
      • 1.3.1 简介
      • 1.3.2 命令
      • 1.3.3 数据结构
    • 1.4 集合(Set)
      • 1.4.1 简介
      • 1.4.2 命令
      • 1.4.3 数据结构
    • 1.5 哈希(Hash)
      • 1.5.1 简介
      • 1.5.2 命令
      • 1.5.3 数据结构
    • 1.6 有序集合(Zset)
      • 1.6.1 简介
      • 1.6.2 命令
      • 1.6.3 数据结构
      • 1.6.4 有序链表与跳跃表
  • 二、配置文件
      • **Units单位**
      • INCLUDES
    • 网络相关配置
      • bind
      • protected-mode
      • PORT
      • tcp-backlog
      • timeout
      • tcp-keepalive
    • 通用配置
      • daemonize
      • pidfile
      • loglevel
      • logfile
      • databases
    • SECURITY安全配置
      • requirepass
    • LIMITS限制
      • maxclients
      • maxmemory
      • maxmemory-policy
      • maxmemory-samples
  • 三、新数据类型
    • 3.1 Bitmaps
      • 3.1.1 简介
      • 3.1.2 命令
    • 3.2 HyperLogLog
      • 3.2.1 简介
      • 3.2.2 命令
    • 3.3 Geospatial
      • 3.3.1 简介
      • 3.3.2 命令
  • 四、Redis事务
    • 4.1 简介
    • 4.2 命令
    • 4.3 事务的错误处理
      • 4.3.1 组队中
      • 4.3.2 执行中
    • 4.4 事务冲突
      • 4.4.1 悲观锁
      • 4.4.2 乐观锁
    • 4.5 监视WATCH
      • 4.5.1 简介
    • 4.6 取消监视 UNWATCH
      • 4.6.1 简介
  • 五、Redis持久化
    • 5.1 RDB持久化
      • 5.1.1 简介
      • 5.1.2 执行流程
      • 5.1.3 fork
      • 5.1.4 配置
        • 5.1.4.1 文件名称配置
        • 5.1.4.2 文件位置配置
        • 5.1.4.3 快照配置
        • 5.1.4.4 save配置
        • 5.1.4.5 stop-writes-on-bgsave-error
        • 5.1.4.6 rdbcompression
        • 5.1.4.7 rdbchecksum
      • 5.1.5 RDB备份
      • 5.1.6 优势与劣势
        • 5.1.6.1 优势
        • 5.1.6.2 劣势
      • 5.1.7 如何停止
    • 5.2 AOF持久化
      • 5.2.1 简介
      • 5.2.2 执行流程
      • 5.2.3 配置
        • 5.2.3.1 开启
        • 5.2.3.2 文件名
        • 5.2.3.3 AOF和RDB同时开启
        • 5.2.3.4 AOF修复
        • 5.2.3.5 同步设置
        • 5.2.3.6 Rewrite压缩
          • 5.2.3.6.1 简介
          • 5.2.3.6.2 原理
          • 5.2.3.6.3 何时重写
          • 5.2.3.6.4 重写流程
      • 5.2.4 优势与劣势
        • 5.2.4.1 优势
        • 5.2.4.2 劣势
    • 5.3 AOF和RDB的选择
  • 六、主从复制和集群
    • 6.1 主从复制
      • 6.1.1 简介
      • 6.1.2 作用
      • 6.1.3 搭建主从
      • 6.1.4 常用三招
        • 6.1.4.1 一主二仆
            • 复制原理
        • 6.1.4.2 薪火相传
        • 6.1.4.3 反客为主
      • 6.1.5 哨兵模式
        • 6.1.5.1 如何配置启动
        • 6.1.5.2 复制延时
        • 6.1.5.3 故障恢复
        • 6.1.5.4 Jedis对象
    • 6.2 集群
      • 6.2.1 简介
      • 6.2.2 集群搭建
      • 6.2.3 集群操作和故障恢复
        • 6.2.3.1 集群操作
        • 6.2.3.2 故障恢复
      • 6.2.4 集群的Jedis开发
      • 6.2.5 Redis的好处与不足
        • 6.2.5.1 好处
        • 6.2.5.2 不足
  • 七、Redis应用问题的解决
    • 7.1 缓存穿透
      • 7.1.1 问题描述
      • 7.1.2 解决方案
    • 7.2 缓存击穿
      • 7.2.1 问题描述
      • 7.2.2 解决方案
    • 7.3 缓存雪崩
      • 7.3.1 问题描述
      • 7.3.2 解决方案
  • 八、Redis6.0的新功能
    • 8.1 ACL
      • 8.1.1 简介
      • 8.1.2 命令
      • 8.1.3 ACL规则
    • 8.2 IO多线程
      • 8.2.1 原理
    • 8.3 工具支持 Cluster
    • 8.4 其他新功能

一、简介

  • Redis作为一种key/value结构的数据存储系统
  • 默认16个数据库,类似数组下标从0开始,初始默认使用0号库
  • 数据类型包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。

1.1 Redis键(key)

作用语法
登陆本地redisredis-cli 或者 redis-cli -p 6379 或者 redis-cli -p 6379 -a password #-a后面为password,此操作需要开启redis.conf文件中的 requirepass选项
登陆远程redisredis-cli -h ip -p 6379 -a password
查看redis信息info
清除redis屏幕内容clear
退出redis服务exit
关闭redis服务shutdown
切换数据库select (0-15)
清除当前数据库数据flushdb
清除所有数据库数据flushall
查看redis中的keykeys *
判断某个key是否存在exists key
查看你的key是什么类型type key
删除指定的key数据del key
根据value选择非阻塞删除 仅将keys从keyspace元数据中删除,真正的删除会在后续异步操作unlink key
10秒钟:为给定的key设置过期时间expire key 10
清除redis中的数据del test1
查看当前数据库的key的数量dbsize
查看还有多少秒过期,-1表示永不过期,-2表示已过期ttl key
取消时长设置PERSIST key

1.2 字符串(String)

1.2.1 简介

  • String是Redis最基本的类型,一个key对应一个value
  • String类型是二进制安全的。意味着Redis的string可以包含任何数据。比如jpg图片或者序列化的对象。
  • String类型是Redis最基本的数据类型,一个Redis中字符串value最多可以是512M

1.2.2 命令

作用语法
添加键值对set
查询对应键值get
向尾部追加值append >
返回数据的长度,如果键不存在则返回0strlen
只有在 key 不存在时 设置 key 的值setnx
将 key 中储存的数字值增1
只能对数字值操作,如果为空,新增值为1
incr >
将 key 中储存的数字值减1
只能对数字值操作,如果为空,新增值为-1
decr
将 key 中储存的数字值增减。自定义步长incrby / decrby <步长>
同时设置一个或多个 key-value对mset
同时获取一个或多个 valuemget
同时设置一个或多个 key-value 对,当且仅当所有给定 key 都不存在。msetnx
获得值的范围,类似java中的substring,前包,后包getrange<起始位置><结束位置>
覆写所储存的字符串值,从<起始位置>开始(索引从0开始)setrange <起始位置>
设置键值的同时,设置过期时间,单位秒。setex <过期时间>
以新换旧,设置了新值同时获得旧值。getset

1.2.3 数据结构

String的数据结构为简单动态字符串(Simple Dynamic String,缩写SDS)。是可以修改的字符串,内部结构实现上类似于Java的ArrayList,采用预分配冗余空间的方式来减少内存的频繁分配.

在这里插入图片描述

如图中所示,内部为当前字符串实际分配的空间capacity一般要高于实际字符串长度len。当字符串长度小于1M时,扩容都是加倍现有的空间,如果超过1M,扩容时一次只会多扩1M的空间。需要注意的是字符串最大长度为512M。

1.3 列表(List)

1.3.1 简介

  • Redis的list类型相当于java中的LinkedList,其原理就就是一个双向链表。支持正向、反向查找和遍历等操作,插入删除速度比较快。单键对应多值。

在这里插入图片描述

1.3.2 命令

作用语法
从左边/右边插入一个或多个值lpush/rpush
从左边/右边吐出n个值。值在键在,值光键亡lpop/rpop
列表右边吐出一个值,插到列表左边rpoplpush
按照索引下标获得元素(从左到右)lrange
0左边第一个,-1右边第一个,(0-1表示获取所有)lrange mylist 0 -1
按照索引下标获得元素(从左到右)lindex
获得列表长度llen
的后面插入插入值linsert before
从左边删除n个value(从左到右)lrem
将列表key下标为index的值替换成valueset

1.3.3 数据结构

List的数据结构为快速链表quickList。

首先在列表元素较少的情况下会使用一块连续的内存存储,这个结构是ziplist,也即是压缩列表。

它将所有的元素紧挨着一起存储,分配的是一块连续的内存。

当数据量比较多的时候才会改成quicklist。

因为普通的链表需要的附加指针空间太大,会比较浪费空间。比如这个列表里存的只是int类型的数据,结构上还需要两个额外的指针prev和next。

在这里插入图片描述

Redis将链表和ziplist结合起来组成了quicklist。也就是将多个ziplist使用双向指针串起来使用。这样既满足了快速的插入删除性能,又不会出现太大的空间冗余。

1.4 集合(Set)

1.4.1 简介

  • Redis set对外提供的功能与list类似是一个列表的功能,特殊之处在于set是可以自动排重的,当你需要存储一个列表数据,又不希望出现重复数据时,set是一个很好的选择,并且set提供了判断某个成员是否在一个set集合内的重要接口,这个也是list所不能提供的。

  • Redis的Set是string类型的无序集合。它底层其实是一个value为null的hash表,所以添加,删除,查找的复杂度都是O(1)。

  • 一个算法,随着数据的增加,执行时间的长短,如果是O(1),数据增加,查找数据的时间不变

1.4.2 命令

作用语法
将一个或多个 member 元素加入到集合 key 中,已经存在的 member 元素将被忽略sadd
获取集合中成员smembers
判断集合是否为含有该值,有1,没有0sismember
删除集合中的某个元素srem
随机从该集合中吐出一个值spop key
获取集合中的成员个数scard
随机从该集合中取出n个值。不会从集合中删除srandmember
把集合中一个值从一个集合移动到另一个集合smove value
返回两个集合的并集元素sunion
返回两个集合的交集元素sinter
返回两个集合的差集元素(key1中的,不包含key2中的)sdiff

1.4.3 数据结构

  • Set数据结构是dict字典,字典是用哈希表实现的。

  • Java中HashSet的内部实现使用的是HashMap,只不过所有的value都指向同一个对象。Redis的set结构也是一样,它的内部也使用hash结构,所有的value都指向同一个内部值。

1.5 哈希(Hash)

1.5.1 简介

  • Redis hash 是一个键值对集合。

  • Redis hash是一个string类型的field和value的映射表,hash特别适合用于存储对象。

类似Java里面的Map

  • Redis散列类型相当于Java中的HashMap,实现原理跟HashMap一致,一般用于存储对象信息,存储了字段(field)和字段值的映射,一个散列类型可以包含最多232-1个字段。

在这里插入图片描述

1.5.2 命令

作用语法
集合中的 键赋值hset
集合取出 valuehget
批量设置hash的值hmset
查看哈希表 key 中,给定域 field 是否存在hexists
列出该hash集合的所有fieldhkeys
列出该hash集合的所有valuehvals
为哈希表 key 中的域 field 的值加上增量 1 -1hincrby
将哈希表 key 中的域 的值设置为 ,当且仅当域 不存在hsetnx

1.5.3 数据结构

  • Hash类型对应的数据结构是两种:ziplist(压缩列表),hashtable(哈希表)。当field-value长度较短且个数较少时,使用ziplist,否则使用hashtable

1.6 有序集合(Zset)

1.6.1 简介

  • Redis有序集合zset与普通集合set非常相似,是一个没有重复元素的字符串集合。

  • 不同之处是有序集合的每个成员都关联了一个评分(score),这个评分(score)被用来按照从最低分到最高分的方式排序集合中的成员。集合的成员是唯一的,但是评分可以是重复了 。

  • 因为元素是有序的, 所以你也可以很快的根据评分(score)或者次序(position)来获取一个范围的元素。

  • 访问有序集合的中间元素也是非常快的,因此你能够使用有序集合作为一个没有重复成员的智能列表。

1.6.2 命令

作用命令
将一个或多个 member 元素及其 score 值加入到有序集 key 当中。zadd
返回有序集 key 中,下标在之间的元素
带WITHSCORES,可以让分数一起和值返回到结果集。
zrange [WITHSCORES]
返回有序集 key 中,所有 score 值介于 min 和 max 之间(包括等于 min 或 max )的成员。有序集成员按 score 值递增(从小到大)次序排列zrangebyscore key minmax [withscores] [limit offset count]
同上,改为从大到小排列zrevrangebyscore key maxmin [withscores] [limit offset count]
为元素的score加上增量zincrby
删除该集合下,指定值的元素zrem
统计该集合,分数区间内的元素个数zcount
返回该值在集合中的排名,从0开始zrank

1.6.3 数据结构

  • SortedSet(zset)是Redis提供的一个非常特别的数据结构,一方面它等价于Java的数据结构Map,可以给每一个元素value赋予一个权重score,另一方面它又类似于TreeSet,内部的元素会按照权重score进行排序,可以得到每个元素的名次,还可以通过score的范围来获取元素的列表。

  • zset底层使用了两个数据结构

​ hash,hash的作用就是关联元素value和权重score,保障元素value的唯一性,可以通过元素value找到相应的score值。

​ 跳跃表,跳跃表的目的在于给元素value排序,根据score的范围获取元素列表。

1.6.4 有序链表与跳跃表

  • 有序链表

在这里插入图片描述

  • 跳跃表

在这里插入图片描述

二、配置文件

Units单位

​ 配置大小单位,开头定义了一些基本的度量单位,只支持bytes,不支持bit,大小写不敏感

在这里插入图片描述

INCLUDES

​ 类似jsp中的include,多实例的情况可以把公用的配置文件提取出来

在这里插入图片描述

网络相关配置

bind

  • 默认情况bind=127.0.0.1只能接受本机的访问请求

  • 不写的情况下,无限制接受任何ip

  • 生产环境肯定要写你应用服务器的地址;服务器是需要远程访问的,所以需要将其注释掉

  • 如果开启了protected-mode,那么在没有设定bind ip且没有设密码的情况下,Redis只允许接受本机的响应

protected-mode

  • 本机访问保护模式设置

PORT

  • 端口号

tcp-backlog

  • 设置tcp的backlog,backlog其实是一个连接队列,backlog队列总和=未完成三次握手队列 + 已经完成三次握手队列。

  • 在高并发环境下你需要一个高backlog值来避免慢客户端连接问题。

  • 注意Linux内核会将这个值减小到/proc/sys/net/core/somaxconn的值(128),所以需要确认增大/proc/sys/net/core/somaxconn和/proc/sys/net/ipv4/tcp_max_syn_backlog(128)两个值来达到想要的效果

timeout

  • 一个空闲的客户端维持多少秒会关闭,0表示关闭该功能。即永不关闭

tcp-keepalive

  • 对访问客户端的一种心跳检测,每个n秒检测一次。

  • 单位为秒,如果设置为0,则不会进行Keepalive检测,建议设置成60

通用配置

daemonize

  • 是否为后台进程,设置为yes

  • 守护进程,后台启动

pidfile

  • 存放pid文件的位置,每个实例会产生一个不同的pid文件

loglevel

  • 指定日志记录级别,Redis总共支持四个级别:debug、verbose、notice、warning,默认为notice

  • 四个级别根据使用阶段来选择,生产环境选择notice 或者warning

logfile

  • 日志文件名称

databases

  • 设定库的数量 默认16,默认数据库为0,可以使用SELECT 命令在连接上指定数据库id

SECURITY安全配置

requirepass

  • 设置密码

LIMITS限制

maxclients

  • 设置redis同时可以与多少个客户端进行连接。

  • 默认情况下为10000个客户端。

  • 如果达到了此限制,redismax number of clients reached

maxmemory

  • 建议必须设置,否则,将内存占满,造成服务器宕机
  • 设置redis可以使用的内存量。一旦到达内存使用上限,redis将会试图移除内部数据,移除规则可以通过maxmemory-policy来指定。
  • 如果redis无法根据移除规则来移除内存中的数据,或者设置了“不允许移除”,那么redis则会针对那些需要申请内存的指令返回错误信息,比如SET、LPUSH等。
  • 但是对于无内存申请的指令,仍然会正常响应,比如GET等。如果你的redis是主redis(说明你的redis有从redis),那么在设置内存使用上限时,需要在系统中留出一些内存空间给同步队列缓存,只有在你设置的是“不移除”的情况下,才不用考虑这个因素。

maxmemory-policy

  • volatile-lru:使用LRU算法移除key,只对设置了过期时间的键;(最近最少使用)
  • allkeys-lru:在所有集合key中,使用LRU算法移除key
  • volatile-random:在过期集合中移除随机的key,只对设置了过期时间的键
  • allkeys-random:在所有集合key中,移除随机的key
  • volatile-ttl:移除那些TTL值最小的key,即那些最近要过期的key
  • noeviction:不进行移除。针对写操作,只是返回错误信息

maxmemory-samples

  • 设置样本数量,LRU算法和最小TTL算法都并非是精确的算法,而是估算值,所以你可以设置样本的大小,redis默认会检查这么多个key并选择其中LRU的那个。

  • 一般设置3到7的数字,数值越小样本越不准确,但性能消耗越小。

三、新数据类型

3.1 Bitmaps

3.1.1 简介

Redis提供了Bitmaps这个“数据类型”可以实现对位的操作:

Bitmaps本身不是一种数据类型, 实际上它就是字符串(key-value) , 但是它可以对字符串的位进行操作。
Bitmaps单独提供了一套命令,所以在中使用和使用字符串的方法不太相同。可以把想象成一个以位为单位的数组,数组的每个单元只能存储0和1,数组的下标在中叫做偏移量
在这里插入图片描述

3.1.2 命令

作用语法
设置Bitmaps中某个偏移量的值(0或1)setbit
获取Bitmaps中某个偏移量的值getbit
统计字符串被设置为1的bit数
额外的 start 或 end 参数,可以让计数只在特定的位上进行
bitcount
bitop是一个复合操作, 它可以做多个Bitmaps的and(交集) 、
or(并集) 、 not(非) 、 xor(异或) 操作并将结果保存在destkey中
bitop and(or/not/xor) [key…]

3.2 HyperLogLog

3.2.1 简介

在工作当中,我们经常会遇到与统计相关的功能需求,比如统计网站PV(PageView页面访问量),可以使用Redis的incr、incrby轻松实现。但像UV(UniqueVisitor,独立访客)、独立IP数、搜索记录数等需要去重和计数的问题如何解决?这种求集合中不重复元素个数的问题称为基数问题。

3.2.2 命令

作用语法
添加指定元素到 HyperLogLog中pfadd < element> [element …]
计算HLL的近似基数,可以计算多个HLL,比如用HLL存储每天的UV,计算一周的UV可以使用7天的UV合并计算即可pfcount [key …]
将一个或多个HLL合并后的结果存储在另一个HLL中,比如每月活跃用户可以使用每天的活跃用户来合并计算可得pfmerge [sourcekey …]

3.3 Geospatial

3.3.1 简介

Redis 3.2 中增加了对GEO类型的支持。GEO,Geographic,地理信息的缩写。该类型,就是元素的2维坐标,在地图上就是经纬度。redis基于该类型,提供了经纬度设置,查询,范围查询,距离查询,经纬度Hash等常见操作。

3.3.2 命令

作用语法
添加地理位置(经度,纬度,名称)geoadd< longitude> [longitude latitude member…]
获得指定地区的坐标值geopos [member…]
获取两个位置之间的直线距离geodist [m
以给定的经纬度为中心,找出某一半径内的元素 四个参数 -》经度 纬度 距离 单位georadius< longitude>radius m

四、Redis事务

4.1 简介

Redis事务是一个单独的隔离操作:事务中的所有命令都会序列化、按顺序地执行。事务在执行的过程中,不会被其他客户端发送来的命令请求所打断。

Redis事务的主要作用就是串联多个命令防止别的命令插队。

4.2 命令

从输入Multi命令开始,输入的命令都会依次进入命令队列中,但不会执行,直到输入Exec后,Redis会将之前的命令队列中的命令依次执行。

组队的过程中可以通过discard。

在这里插入图片描述

4.3 事务的错误处理

4.3.1 组队中

组队中某个命令出现了报告错误,执行时整个的所有队列都会被取消

在这里插入图片描述

4.3.2 执行中

如果执行阶段某个命令报出了错误,则只有报错的命令不会被执行,而其他的命令都会执行,不会回滚。

在这里插入图片描述

4.4 事务冲突

4.4.1 悲观锁

悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。
在这里插入图片描述

4.4.2 乐观锁

乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量。Redis就是利用这种check-and-set机制实现事务的。
在这里插入图片描述

4.5 监视WATCH

4.5.1 简介

在执行multi之前,先执行watch key1 [key2],可以监视一个(或多个) key ,如果在事务执行之前这个(或这些) key 被其他命令所改动,那么事务将被打断。

4.6 取消监视 UNWATCH

4.6.1 简介

取消 WATCH 命令对所有 key 的监视。

如果在执行 WATCH 命令之后,EXEC 命令或DISCARD 命令先被执行了的话,那么就不需要再执行UNWATCH了

五、Redis持久化

5.1 RDB持久化

5.1.1 简介

在指定的时间间隔内将内存中的数据集快照写入磁盘, 也就是行话讲的Snapshot快照,它恢复时是将快照文件直接读到内存里

5.1.2 执行流程

Redis会单独创建(fork)一个子进程来进行持久化,会先将数据写入到 一个临时文件中,待持久化过程都结束了,再用这个临时文件替换上次持久化好的文件。 整个过程中,主进程是不进行任何IO操作的,这就确保了极高的性能 如果需要进行大规模数据的恢复,且对于数据恢复的完整性不是非常敏感,那RDB方式要比AOF方式更加的高效。RDB的缺点是最后一次持久化后的数据可能丢失。
在这里插入图片描述

5.1.3 fork

  • Fork的作用是复制一个与当前进程一样的进程。新进程的所有数据(变量、环境变量、程序计数器等) 数值都和原进程一致,但是是一个全新的进程,并作为原进程的子进程
  • 在Linux程序中,fork()会产生一个和父进程完全相同的子进程,但子进程在此后多会exec系统调用,出于效率考虑,Linux中引入了“写时复制技术”
  • 一般情况父进程和子进程会共用同一段物理内存,只有进程空间的各段的内容要发生变化时,才会将父进程的内容复制一份给子进程。

5.1.4 配置

5.1.4.1 文件名称配置

​ 在redis.conf中配置文件名称,默认为dump.rdb

在这里插入图片描述

5.1.4.2 文件位置配置

​ 配置文件的位置,默认是启动位置

在这里插入图片描述

5.1.4.3 快照配置

在这里插入图片描述

5.1.4.4 save配置

格式:save 秒钟 写操作次数

5.1.4.5 stop-writes-on-bgsave-error

当Redis无法写入磁盘的话,直接关掉Redis的写操作。推荐yes

5.1.4.6 rdbcompression

对于存储到磁盘中的快照,可以设置是否进行压缩存储。如果是的话,redis会采用LZF算法进行压缩。

如果你不想消耗CPU来进行压缩的话,可以设置为关闭此功能。推荐yes.

5.1.4.7 rdbchecksum

检查完整性

在存储快照后,还可以让redis使用CRC64算法来进行数据校验,

但是这样做会增加大约10%的性能消耗,如果希望获取到最大的性能提升,可以关闭此功能->推荐yes

5.1.5 RDB备份

先通过config get dir 查询rdb文件的目录

将*.rdb的文件拷贝到别的地方

rdb的恢复

  • 关闭Redis
  • 先把备份的文件拷贝到工作目录下 cp dump2.rdb dump.rdb
  • 启动Redis,备份数据会直接加载

5.1.6 优势与劣势

5.1.6.1 优势

  • 适合大规模的数据恢复
  • 对数据完整性和一致性要求不高更适合使用
  • 节省磁盘空间
  • 恢复速度快

在这里插入图片描述

5.1.6.2 劣势

  • Fork的时候,内存中的数据被克隆了一份,大致2倍的膨胀性需要考虑
  • 虽然Redis在fork时使用了写时拷贝技术,但是如果数据庞大时还是比较消耗性能。
  • 在备份周期在一定间隔时间做一次备份,所以如果Redis意外down掉的话,就会丢失最后一次快照后的所有修改。

5.1.7 如何停止

动态停止RDB:redis-cli config set save “” #save后给空值,表示禁用保存策略

在这里插入图片描述

5.2 AOF持久化

5.2.1 简介

日志的形式来记录每个写操作(增量保存),将Redis执行过的所有写指令记录下来(读操作不记录), 只许追加文件但不可以改写文件,redis启动之初会读取该文件重新构建数据,换言之,redis 重启的话就根据日志文件的内容将写指令从前到后执行一次以完成数据的恢复工作

5.2.2 执行流程

  • 客户端的请求写命令会被append追加到AOF缓冲区内;

  • AOF缓冲区根据AOF持久化策略[always,everysec,no]将操作sync同步到磁盘的AOF文件中;

  • AOF文件大小超过重写策略或手动重写时,会对AOF文件rewrite重写,压缩AOF文件容量;

  • Redis服务重启时,会重新load加载AOF文件中的写操作达到数据恢复的目的;

在这里插入图片描述

5.2.3 配置

5.2.3.1 开启

默认的appendonly no,改为yes

5.2.3.2 文件名

可以在redis.conf中配置文件名称,默认为 appendonly.aof

AOF文件的保存路径,同RDB路径一致.

5.2.3.3 AOF和RDB同时开启

AOF和RDB同时开启,系统默认取AOF的数据(数据不会存在丢失)

5.2.3.4 AOF修复

  • 如遇到AOF文件损坏,通过/usr/local/bin/redis-check-aof–fix appendonly.aof进行恢复
  • 备份被写坏的AOF文件
  • 恢复:重启redis,然后重新加载

5.2.3.5 同步设置

  • appendfsync always

始终同步,每次Redis的写入都会立刻记入日志;性能较差但数据完整性比较好

  • appendfsync everysec

每秒同步,每秒记入日志一次,如果宕机,本秒的数据可能丢失。

  • appendfsync no

redis不主动进行同步,把同步时机交给操作系统。

5.2.3.6 Rewrite压缩

5.2.3.6.1 简介

AOF采用文件追加方式,文件会越来越大为避免出现此种情况,新增了重写机制, 当AOF文件的大小超过所设定的阈值时,Redis就会启动AOF文件的内容压缩, 只保留可以恢复数据的最小指令集.可以使用命令bgrewriteaof.

5.2.3.6.2 原理

AOF文件持续增长而过大时,会fork出一条新进程来将文件重写(也是先写临时文件最后再rename),redis4.0版本后的重写,是指上就是把rdb 的快照,以二级制的形式附在新的aof头部,作为已有的历史数据,替换掉原来的流水账操作。

no-appendfsync-on-rewrite:

如果 no-appendfsync-on-rewrite=yes ,不写入aof文件只写入缓存,用户请求不会阻塞,但是在这段时间如果宕机会丢失这段时间的缓存数据。(降低数据安全性,提高性能)

如果 no-appendfsync-on-rewrite=no, 还是会把数据往磁盘里刷,但是遇到重写操作,可能会发生阻塞。(数据安全,但是性能降低)

5.2.3.6.3 何时重写

Redis会记录上次重写时的AOF大小,默认配置是当AOF文件大小是上次rewrite后大小的一倍且文件大于64M时触发

重写虽然可以节约大量磁盘空间,减少恢复时间。但是每次重写还是有一定的负担的,因此设定Redis要满足一定条件才会进行重写。

auto-aof-rewrite-percentage:设置重写的基准值,文件达到100%时开始重写(文件是原来重写后文件的2倍时触发)

auto-aof-rewrite-min-size:设置重写的基准值,最小文件64MB。达到这个值开始重写。

5.2.3.6.4 重写流程

bgrewriteaof触发重写,判断是否当前有bgsave或bgrewriteaof在运行,如果有,则等待该命令结束后再继续执行。
主进程fork出子进程执行重写操作,保证主进程不会阻塞。
子进程遍历redis内存中数据到临时文件,客户端的写请求同时写入aof_buf缓冲区和aof_rewrite_buf重写缓冲区保证原AOF文件完整以及新AOF文件生成期间的新的数据修改动作不会丢失。
1).子进程写完新的AOF文件后,向主进程发信号,父进程更新统计信息。2).主进程把aof_rewrite_buf中的数据写入到新的AOF文件。
使用新的AOF文件覆盖旧的AOF文件,完成AOF重写。
在这里插入图片描述

5.2.4 优势与劣势

5.2.4.1 优势

  • 备份机制更稳健,丢失数据概率更低。
  • 可读的日志文本,通过操作AOF

在这里插入图片描述

5.2.4.2 劣势

  • 比起RDB占用更多的磁盘空间。
  • 恢复备份速度要慢。
  • 每次读写都同步的话,有一定的性能压力。
  • 存在个别Bug,造成恢复不能。

在这里插入图片描述

5.3 AOF和RDB的选择

官方推荐两个都启用。

如果对数据不敏感,可以选单独用RDB。

不建议单独用 AOF,因为可能会出现Bug。

如果只是做纯内存缓存,可以都不用。

RDB持久化方式能够在指定的时间间隔能对你的数据进行快照存储

AOF持久化方式记录每次对服务器写的操作,当服务器重启的时候会重新执行这些命令来恢复原始的数据,AOF命令以redis协议追加保存每次写的操作到文件末尾.

Redis还能对AOF文件进行后台重写,使得AOF文件的体积不至于过大

只做缓存:如果你只希望你的数据在服务器运行的时候存在,你也可以不使用任何持久化方式.

同时开启两种持久化方式

在这种情况下,当redis重启的时候会优先载入AOF文件来恢复原始的数据, 因为在通常情况下AOF文件保存的数据集要比RDB文件保存的数据集要完整.

RDB的数据不实时,同时使用两者时服务器重启也只会找AOF文件。那要不要只使用AOF呢?

建议不要,因为RDB更适合用于备份数据库(AOF在不断变化不好备份), 快速重启,而且不会有AOF可能潜在的bug,留着作为一个万一的手段。

性能建议

因为RDB文件只用作后备用途,建议只在Slave上持久化RDB文件,而且只要15分钟备份一次就够了,只保留save 900 1这条规则。

如果使用AOF,好处是在最恶劣情况下也只会丢失不超过两秒数据,启动脚本较简单只load自己的AOF文件就可以了。

代价,一是带来了持续的IO,二是AOF rewrite的最后将rewrite过程中产生的新数据写到新文件造成的阻塞几乎是不可避免的。

只要硬盘许可,应该尽量减少AOF rewrite的频率,AOF重写的基础大小默认值64M太小了,可以设到5G以上。

默认超过原大小100%大小时重写可以改到适当的数值。

六、主从复制和集群

6.1 主从复制

6.1.1 简介

主机数据更新后根据配置和策略, 自动同步到备机的master/slaver机制,Master以写为主,Slave以读为主

6.1.2 作用

  • 读写分离,性能扩展
  • 容灾快速恢复

在这里插入图片描述

6.1.3 搭建主从

  • 在根目录下创建文件夹myredis,把redis的配置文件复制过来,要把aof持久化关掉
  • 创建三个文件redis6379.conf、redis6381.conf、redis6380.conf内容如下
######################redis6379.conf#######################
include /myredis/redis.conf
pidfile /var/run/redis_6379.pid
port 6379
dbfilename dump6379.rdb######################redis6380.conf#######################
include /myredis/redis.conf
pidfile /var/run/redis_6380.pid
port 6380
dbfilename dump6380.rdb######################redis6381.conf#######################
include /myredis/redis.conf
pidfile /var/run/redis_6381.pid
port 6381
dbfilename dump6381.rdb
  • 查看三台主机运行情况
info replication  # 打印主从复制的相关信息
  • 配从(库)不配主(库)
slaveof    # 在从机上配置主机的ip和端口 在6380和6381上执行: slaveof 127.0.0.1 6379
  • 主机挂掉,重启就行,一切如初,从机重启需重设:slaveof 127.0.0.1 6379

6.1.4 常用三招

6.1.4.1 一主二仆

  • slave1、slave2是从头开始复制还是从切入点开始复制?比如从k4进来,那之前的k1,k2,k3是否也可以复制?

​ 从机会全量复制主机的内容,k1,k2,k3,k4都会复制

  • 从机是否可以写?set可否?

​ 从机只可读,不可写

  • 主机shutdown后情况如何?从机是上位还是原地待命?

​ 主机shutdown后,从机原地待命,等待主机重新启动,一切回复正常

  • 主机又回来了后,主机新增记录,从机还能否顺利复制?

​ 可以复制,因为主机重启后和之前一样,主机写内容,会同步到从机中

  • 其中一台从机down后情况如何?依照原有它能跟上大部队吗?

    从机down后,会脱离大部队,如果重新启动,还想同步主机内容的话,需要重新执行命令

    slaveof

复制原理

Slave启动成功连接到master后会发送一个sync命令

Master接到命令启动后台的存盘进程,同时收集所有接收到的用于修改数据集命令, 在后台进程执行完毕之后,master将传送整个数据文件到slave,以完成一次完全同步

全量复制:而slave服务在接收到数据库文件数据后,将其存盘并加载到内存中。

增量复制:Master继续将新的所有收集到的修改命令依次传给slave,完成同步

但是只要是重新连接master,一次完全同步(全量复制)将被自动执行
在这里插入图片描述

6.1.4.2 薪火相传

上一个Slave可以是下一个slave的Master,Slave同样可以接收其他 slaves的连接和同步请求,那么该slave作为了链条中下一个的master, 可以有效减轻master的写压力,去中心化降低风险。

用 slaveof

中途变更转向:会清除之前的数据,重新建立拷贝最新的

风险是一旦某个slave宕机,后面的slave都没法备份

主机挂了,从机还是从机,无法写数据了

6.1.4.3 反客为主

当一个master宕机后,后面的slave可以立刻升为master,其后面的slave不用做任何修改。

用slaveof no one 将从机变为主机

6.1.5 哨兵模式

反客为主的自动版,能够后台监控主机是否故障,如果故障了根据投票数自动将从库转换为主库

6.1.5.1 如何配置启动

  • 先搭建一主二从的环境
  • 自定义的/myredis目录下新建sentinel.conf文件,名字绝不能错
  • 配置哨兵,填写内容
sentinel monitor mymaster 127.0.0.1 6379 1
#其中mymaster为监控对象起的服务器名称, 1 为至少有多少个哨兵同意迁移的数量。
  • 启动哨兵,执行redis-sentinel /myredis/sentinel.conf

  • 当主机挂掉,从机选举中产生新的主机

  • 重新启动原主机,原主机重启后会变为从机

6.1.5.2 复制延时

由于所有的写操作都是先在Master上操作,然后同步更新到Slave上,所以从Master同步到Slave机器有一定的延迟,当系统很繁忙的时候,延迟问题会更加严重,Slave机器数量的增加也会使这个问题更加严重。

6.1.5.3 故障恢复

优先级在redis.conf中默认:slave-priority 100,值越小优先级越高

偏移量是指获得原主机数据最全的

每个redis实例启动后都会随机生成一个40位的runid

在这里插入图片描述

6.1.5.4 Jedis对象

private static JedisSentinelPool jedisSentinelPool=null;public static  Jedis getJedisFromSentinel(){if(jedisSentinelPool==null){Set sentinelSet=new HashSet<>();sentinelSet.add("ip:端口");JedisPoolConfig jedisPoolConfig =new JedisPoolConfig();jedisPoolConfig.setMaxTotal(10); //最大可用连接数jedisPoolConfig.setMaxIdle(5); //最大闲置连接数jedisPoolConfig.setMinIdle(5); //最小闲置连接数jedisPoolConfig.setBlockWhenExhausted(true); //连接耗尽是否等待jedisPoolConfig.setMaxWaitMillis(2000); //等待时间jedisPoolConfig.setTestOnBorrow(true); //取连接的时候进行一下测试 ping pongjedisSentinelPool=new JedisSentinelPool("mymaster",sentinelSet,jedisPoolConfig);return jedisSentinelPool.getResource();}else{return jedisSentinelPool.getResource();}
}

6.2 集群

6.2.1 简介

Redis 集群实现了对Redis的水平扩容,即启动N个redis节点,将整个数据库分布存储在这N个节点中,每个节点存储总数据的1/N。

Redis 集群通过分区(partition)来提供一定程度的可用性(availability): 即使集群中有一部分节点失效或者无法进行通讯, 集群也可以继续处理命令请求。

之前通过代理主机来解决,但是redis3.0中提供了解决方案。就是无中心化集群配置。

在这里插入图片描述

6.2.2 集群搭建

  • 删除文件夹中的全部持久化文件rdb或者aof
  • 新建六个配置文件,内容如下:(除了端口号不一样,其他都一样)
include /myredis/redis.conf
pidfile "/var/run/redis_6391.pid"
port 6391
dbfilename "dump6391.rdb"
cluster-enabled yes
cluster-config-file nodes-6391.conf
cluster-node-timeout 15000

cluster-enabled yes 打开集群模式

cluster-config-file nodes-6379.conf 设定节点配置文件名

cluster-node-timeout 15000 设定节点失联时间,超过该时间(毫秒),集群自动进行主从切换。

其中 :%s/6379/6380 是vim的替换命令

  • 启动6个服务

要确保nodes-xxxx.conf生成

  • 将六个节点合成一个集群

组合之前,请确保所有redis实例启动后,nodes-xxxx.conf文件都生成正常

先到redis的src目录中

cd  /opt/redis-6.2.1/src

运行集成集群命令

redis-cli --cluster create --cluster-replicas 1 192.168.242.110:6379 192.168.242.110:6380 192.168.242.110:6381 192.168.242.110:6389 192.168.242.110:6390 192.168.242.110:6391

说明:ip一定要真实ip,不能是localhost或者127.0.0.1

–replicas 1 采用最简单的方式配置集群,一台主机,一台从机,正好三组。

  • 查看是否集成成功
# 连接Redis
redis-cli -c -p 6379
# 查看集群信息
cluster nodes

6.2.3 集群操作和故障恢复

6.2.3.1 集群操作

  • 查看集群信息
cluster nodes
  • redis cluster 如何分配这六个节点

一个集群至少要有三个主节点。

选项 --cluster-replicas 1 表示我们希望为集群中的每个主节点创建一个从节点。

分配原则尽量保证每个主数据库运行在不同的IP地址,每个从库和主库不在一个IP地址上。

  • 什么是slots

在运行集成集群命令后,会出现“[OK] All 16384 slots covered ”

说明:一个 Redis 集群包含 16384 个插槽(hash slot), 数据库中的每个键都属于这 16384 个插槽的其中一个, 集群使用公式CRC16(key) %16384 来计算键key属于哪个槽.其中 CRC16(key) 语句用于计算键 key 的 CRC16 校验和 。

集群中的每个节点负责处理一部分插槽。 举个例子, 如果一个集群可以有主节点, 其中:

节点 A 负责处理 0 号至 5460 号插槽。

节点 B 负责处理 5461 号至 10922 号插槽。

节点 C 负责处理 10923 号至 16383 号插槽。

  • 在集群中录入值

在这里插入图片描述

注意:在用mset 同时设置多个值的时候,需要把这些key放到同一个组中,不然会报错。可以通过{}来定义组的概念,从而使key中{}内相同内容的键值对放到一个slot中去

在这里插入图片描述

  • 查询集群中的值
CLUSTER KEYSLOT k1 # 查询k1的插槽值
CLUSTER COUNTKEYSINSLOT 12706 # 查看指定插槽中的key数量,注意只能在插槽值所在的主机上能成功,例如:12706插槽在6381端口的主机上,但是在其他端口则查询失败
CLUSTER GETKEYSINSLOT 5474 2 # 返回指定插槽的指定数量的key

6.2.3.2 故障恢复

如果主节点下线?从节点能否自动升为主节点?注意:15秒超时

在这里插入图片描述

主节点恢复后,主从关系会如何?主节点回来变成从机。

在这里插入图片描述

如果所有某一段插槽的主从节点都宕掉,redis服务是否还能继续?

如果某一段插槽的主从都挂掉,而cluster-require-full-coverage 为yes ,那么 ,整个集群都挂掉

如果某一段插槽的主从都挂掉,而cluster-require-full-coverage 为no ,那么,该插槽数据全都不能使用,也无法存储。

redis.conf中的参数 cluster-require-full-coverage

6.2.4 集群的Jedis开发

即使连接的不是主机,集群会自动切换主机存储。主机写,从机读。

无中心化主从集群。无论从哪台主机写的数据,其他主机上都能读到数据。

6.2.5 Redis的好处与不足

6.2.5.1 好处

  • 实现扩容
  • 分摊压力
  • 无中心配置相对简单

6.2.5.2 不足

  • 多键操作是不被支持的
  • 多键的Redis事务是不被支持的。lua脚本不被支持
  • 由于集群方案出现较晚,很多公司已经采用了其他的集群方案,而代理或者客户端分片的方案想要迁移至redis cluster,需要整体迁移而不是逐步过渡,复杂度较大。

七、Redis应用问题的解决

7.1 缓存穿透

7.1.1 问题描述

key对应的数据在数据源并不存在,每次针对此key的请求从缓存获取不到,请求都会压到数据源,从而可能压垮数据源。比如用一个不存在的用户id获取用户信息,不论缓存还是数据库都没有,若黑客利用此漏洞进行攻击可能压垮数据库。

在这里插入图片描述

7.1.2 解决方案

一个一定不存在缓存及查询不到的数据,由于缓存是不命中时被动写的,并且出于容错考虑,如果从存储层查不到数据则不写入缓存,这将导致这个不存在的数据每次请求都要到存储层去查询,失去了缓存的意义。

  • 对空值缓存:如果一个查询返回的数据为空(不管是数据是否不存在),我们仍然把这个空结果(null)进行缓存,设置空结果的过期时间会很短,最长不超过五分钟

  • 设置可访问的名单(白名单):使用bitmaps类型定义一个可以访问的名单,名单id作为bitmaps的偏移量,每次访问和bitmap里面的id进行比较,如果访问id不在bitmaps里面,进行拦截,不允许访问。、

  • 采用布隆过滤器:(布隆过滤器(Bloom Filter)是1970年由布隆提出的。它实际上是一个很长的二进制向量(位图)和一系列随机映射函数(哈希函数)。布隆过滤器可以用于检索一个元素是否在一个集合中。它的优点是空间效率和查询时间都远远超过一般的算法,缺点是有一定的误识别率和删除困难。

  • 将所有可能存在的数据哈希到一个足够大的bitmaps中,一个一定不存在的数据会被 这个bitmaps拦截掉,从而避免了对底层存储系统的查询压力。

  • 进行实时监控:当发现的命中率开始急速降低,需要排查访问对象和访问的数据,和运维人员配合,可以设置黑名单限制服务

7.2 缓存击穿

7.2.1 问题描述

key对应的数据存在,但在redis中过期,此时若有大量并发请求过来,这些请求发现缓存过期一般都会从后端DB加载数据并回设到缓存,这个时候大并发的请求可能会瞬间把后端DB压垮。

在这里插入图片描述

7.2.2 解决方案

key可能会在某些时间点被超高并发地访问,是一种非常“热点”的数据。这个时候,需要考虑一个问题:缓存被“击穿”的问题。

  • 预先设置热门数据:在redis高峰访问之前,把一些热门数据提前存入到redis里面,加大这些热门数据key的时长
  • 实时调整:现场监控哪些数据热门,实时调整key的过期时长
  • 使用锁

(1)就是在缓存失效的时候(判断拿出来的值为空),不是立即去load db。

(2) 先使用缓存工具的某些带成功操作返回值的操作(比如Redis的SETNX)去set一个mutex key

(3) 当操作返回成功时,再进行load db的操作,并回设缓存,最后删除mutex key;

(4) 当操作返回失败,证明有线程在load db,当前线程睡眠一段时间再重试整个get缓存的方法。

在这里插入图片描述

7.3 缓存雪崩

7.3.1 问题描述

key对应的数据存在,但在redis中过期,此时若有大量并发请求过来,这些请求发现缓存过期一般都会从后端DB加载数据并回设到缓存,这个时候大并发的请求可能会瞬间把后端DB压垮。

缓存雪崩与缓存击穿的区别在于这里针对很多key缓存,前者则是某一个key正常访问

在这里插入图片描述

7.3.2 解决方案

  • 构建多级缓存架构:nginx缓存 + redis缓存 +其他缓存(ehcache等)

  • 使用锁或队列:用加锁或者队列的方式保证来保证不会有大量的线程对数据库一次性进行读写,从而避免失效时大量的并发请求落到底层存储系统上。不适用高并发情况

  • 设置过期标志更新缓存:记录缓存数据是否过期(设置提前量),如果过期会触发通知另外的线程在后台去更新实际key的缓存。

  • 将缓存失效时间分散开:比如我们可以在原有的失效时间基础上增加一个随机值,比如1-5

八、Redis6.0的新功能

8.1 ACL

8.1.1 简介

Redis ACL是Access Control List(访问控制列表)的缩写,该功能允许根据可以执行的命令和可以访问的键来限制某些连接。

在Redis 5版本之前,Redis 安全规则只有密码控制 还有通过rename 来调整高危命令比如 flushdb , KEYS* , shutdown 等。Redis 6 则提供ACL的功能对用户进行更细粒度的权限控制:

(1)接入权限:用户名和密码

(2)可以执行的命令

(3)可以操作的 KEY

8.1.2 命令

作用语法
展现用户权限列表acl list
查看添加权限指令类别
加参数类型名可以查看类型下具体命令
acl cat
查看当前用户acl whoami
创建和编辑用户ACLacl setuser

8.1.3 ACL规则

类型参数说明
启动和禁用用户on激活某用户账号
启动和禁用用户off禁用某用户账号。注意,已验证的连接仍然可以工作。如果默认用户被标记为off,则新连接将在未进行身份验证的情况下启动,并要求用户使用AUTH选项发送AUTH或HELLO,以便以某种方式进行身份验证。
权限的添加删除+将指令添加到用户可以调用的指令列表中
权限的添加删除-从用户可执行指令列表移除指令
权限的添加删除+@添加该类别中用户要调用的所有指令,有效类别为@admin、@set、@sortedset…等,通过调用ACL CAT命令查看完整列表。特殊类别@all表示所有命令,包括当前存在于服务器中的命令,以及将来将通过模块加载的命令
权限的添加删除-@从用户可调用指令中移除类别
权限的添加删除allcommands+@all的别名
权限的添加删除nocommand-@all的别名
可操作键的添加或删除~ 添加可作为用户可操作的键的模式。例如~*允许所有的键

8.2 IO多线程

Redis6终于支撑多线程了,告别单线程了吗?

IO多线程其实指客户端交互部分网络IO交互处理模块多线程,而非执行命令多线程。Redis6执行命令依然是单线程

8.2.1 原理

Redis 6 加入多线程,但跟 Memcached 这种从 IO处理到数据访问多线程的实现模式有些差异。Redis 的多线程部分只是用来处理网络数据的读写和协议解析,执行命令仍然是单线程。之所以这么设计是不想因为多线程而变得复杂,需要去控制 key、lua、事务,LPUSH/LPOP 等等的并发问题。整体的设计大体如下
在这里插入图片描述

另外,多线程IO默认也是不开启的,需要再配置文件中配置

io-threads-do-reads yes

io-threads 4

8.3 工具支持 Cluster

之前老版Redis想要搭集群需要单独安装ruby环境,Redis 5 将 redis-trib.rb 的功能集成到 redis-cli 。另外官方 redis-benchmark 工具开始支持 cluster 模式了,通过多线程的方式对多个分片进行压测压。

在这里插入图片描述

8.4 其他新功能

Redis6新功能还有:

1、RESP3新的 Redis 通信协议:优化服务端与客户端之间通信

2、Client side caching客户端缓存:基于 RESP3 协议实现的客户端缓存功能。为了进一步提升缓存的性能,将客户端经常访问的数据cache到客户端。减少TCP网络交互。

3、Proxy集群代理模式:Proxy 功能,让 Cluster 拥有像单实例一样的接入方式,降低大家使用cluster的门槛。不过需要注意的是代理不改变 Cluster 的功能限制,不支持的命令还是不会支持,比如跨 slot 的多Key操作。

4、Modules API:Redis 6中模块开发进展非常大,因为为了开发复杂的功能,从一开始就用上模块。可以变成一个框架,利用来构建不同系统,而不需要从头开始写然后还要许可。一开始就是一个向编写各种系统开放的平台。

相关内容

热门资讯

Python|位运算|数组|动... 目录 1、只出现一次的数字(位运算,数组) 示例 选项代...
张岱的人物生平 张岱的人物生平张岱(414年-484年),字景山,吴郡吴县(今江苏苏州)人。南朝齐大臣。祖父张敞,东...
西游西后传演员女人物 西游西后传演员女人物西游西后传演员女人物 孙悟空 六小龄童 唐僧 徐少华 ...
名人故事中贾岛作诗内容简介 名人故事中贾岛作诗内容简介有一次,贾岛骑驴闯了官道.他正琢磨着一句诗,名叫《题李凝幽居》全诗如下:闲...
和男朋友一起优秀的文案? 和男朋友一起优秀的文案?1.希望是惟一所有的人都共同享有的好处;一无所有的人,仍拥有希望。2.生活,...
戴玉手镯的好处 戴玉手镯好还是... 戴玉手镯的好处 戴玉手镯好还是碧玺好 女人戴玉?戴玉好还是碧玺好点佩戴手镯,以和田玉手镯为佳!相嫌滑...
依然什么意思? 依然什么意思?依然(汉语词语)依然,汉语词汇。拼音:yī    rán基本解释:副词,指照往常、依旧...
高尔基的散文诗 高尔基的散文诗《海燕》、《大学》、《母亲》、《童年》这些都是比较出名的一些代表作。
心在飞扬作者简介 心在飞扬作者简介心在飞扬作者简介如下。根据相关公开资料查询,心在飞扬是一位优秀的小说作者,他的小说作...
卡什坦卡的故事赏析? 卡什坦卡的故事赏析?讲了一只小狗的故事, 我也是近来才读到这篇小说. 作家对动物的拟人描写真是惟妙...
林绍涛为简艾拿绿豆糕是哪一集 林绍涛为简艾拿绿豆糕是哪一集第三十二集。 贾宽认为是阎帅间接导致刘映霞住了院,第二天上班,他按捺不...
小爱同学是女生吗小安同学什么意... 小爱同学是女生吗小安同学什么意思 小爱同学,小安同学说你是女生。小安是男的。
内分泌失调导致脸上长斑,怎么调... 内分泌失调导致脸上长斑,怎么调理内分泌失调导致脸上长斑,怎么调理先调理内分泌,去看中医吧,另外用好的...
《魔幻仙境》刺客,骑士人物属性... 《魔幻仙境》刺客,骑士人物属性加点魔幻仙境骑士2功1体质
很喜欢她,该怎么办? 很喜欢她,该怎么办?太冷静了!! 太理智了!爱情是需要冲劲的~不要考虑着考虑那~否则缘...
言情小说作家 言情小说作家我比较喜欢匪我思存的,很虐,很悲,还有梅子黄时雨,笙离,叶萱,还有安宁的《温暖的玄》 小...
两个以名人的名字命名的风景名胜... 两个以名人的名字命名的风景名胜?快太白楼,李白。尚志公园,赵尚志。
幼儿教育的代表人物及其著作 幼儿教育的代表人物及其著作卡尔威特的《卡尔威特的教育》,小卡尔威特,他儿子成了天才后写的《小卡尔威特...
海贼王中为什么说路飞打凯多靠霸... 海贼王中为什么说路飞打凯多靠霸气升级?凯多是靠霸气升级吗?因为之前刚到时确实打不过人家因为路飞的实力...
运气不好拜财神有用吗运气不好拜... 运气不好拜财神有用吗运气不好拜财神有没有用1、运气不好拜财神有用。2、拜财神上香前先点蜡烛,照亮人神...