Redis基础语法和SpringBoot集成使用

news2025/1/4 16:51:27

在初期,已经讲述了Redis安装问题。现在正式进入Redis的入门阶段。

Redis客户端

命令行客户端

redis-cli [options] [commands]

常用到的 options 有:

  • -h 127.0.0.1: 指定要连接的Redis的IP地址【默认127.0.0.1】
  • -p 6379: 指定连接Redis的端口【默认6379】
  • -a 123456: 输入Redis的链接密码

commands 就是操作 Redis 的命令

  • ping: 与 Redis 服务端做心跳测试,服务端会正常返回 PONG
    在这里插入图片描述

设置了密码之后就需要通过密码验证才能使用redis命令行客户端

图形化界面客户端

用的是csdn的github加速计划,所以不用担心无法访问问题
Win免费下载使用Resp.app
在这里插入图片描述
Mac需要付费订阅下载Resp,app
在这里插入图片描述

在这里插入图片描述

Redis常用基础命令

常用命令不需要记忆,需要的时候查询手册即可

Redis通用命令

通用的命令常见的有

  • KEYS:查看符合模板的所有key
  • DEL:删除一个指定的key
  • EXISTS:判断key是否存在
  • EXPIRE:给一个key设置有效期,有效期到期时该key会被自动删除
  • TTL:查看一个KEY的剩余有效期

通过 help 命令帮助这些命令的使用
在这里插入图片描述

String类型

String字符串是Redis最简单的存储类型。
根据字符串格式不同,可以分为3类

  • String:普通字符串
  • int:整形,可自增、自减操作
  • float:整形,可自增、自减操作
    不管是哪种格式,底层都是字节数组形式存储,只不过是编码方式不同。字符串类型的最大空间不能超过512MB
KEYVALUE
name张三
num1
price1.1

常用语法

语法含义
SET添加/修改 已经存在的一个String 类型的键值对
GET根据 KEY 获取 VALUE
MSET批量添加多个 String 类型键值对
MGET根据多个 KEY 获取多个 String 类型的 VALUE
INCR让一个整形的 KEY 自增/自减
INCRBY让一个整形的 KEY 自增指定大小【INCRBY num -2:num -= 2】
INCRBYFLOAT让一个浮点型数据自增
SETNX添加一个 String 类型键值对,前提是这个 KEY 不存在,否则不执行
SETEX添加一个 String 类型键值对,并且指定有效期

KEY 结构

Redis没有MySQL中Table表的概念。如何区分不同类型的 KEY 呢?
比如存储一个ID都为1的用户数据和文章数据,那么 SET ID 1 就会冲突。解决方案是:多个单词之间用 : 分隔开,格式如下:

项目名:业务名:类型:id

user相关的key:BlogSystem:user:1
文章相关的key:BlogSystem:article:1

如果VALUE是一个对象,则可以将对象序列化为JSON字符串后存储

KEYVALUE
BlogSystem:user:1{“id”:1, “name”: “张三”, “age”:13}
BlogSystem:article:1“id”:1, “title”: “Redis快速入门”, “updateTime”: “2022-12-23”

在这里插入图片描述

Hash类型

String结构将对象序列化为JSON格式存储后,当需要修改某个字段是很不方便。
Hash结构可以将对象字段单独存储,方便修改

KEYFILEDVALUE
BlogSystem:user:1name“张三”
BlogSystem:user:1age13
BlogSystem:user:2name“李四”
BlogSystem:user:2age14

Hash常用语法

语法含义
HSET key field value添加或者修改hash类型key的field的值
HGET key field获取一个hash类型key的value
HMSET批量添加多个hash类型key的field的值
HMGET批量获取多个hash类型key的field的值
HGETALL获取一个hash类型的key中的所有的field和value
HKEYS获取一个hash类型的key中所有的field
HINCRBY让一个hash类型key的value自增指定步长
HSETNX添加一个hash类型的key的field之,前提是这个field不存在否则不执行

List类型

Redis中的List类型与Java中的LinkedList类似,可以看作是一个双向链表结构。既可以支持正向检索也支持反向检索。
特征与LinkedList类似:

  • 有序
  • 元素可以重复
  • 插入和删除快
  • 查询一般

用来存储一个有序数据。例如:朋友圈点赞列表,评论列表

常用语法

语法含义
LPUSH key element在列表左侧插入一个或多个元素
RPUSH key element向列表右侧插入一个或多个元素
LPOP key移除并返回列表左侧的第一个元素,没有则返回nil
RPOP key移除并返回列表右侧的第一个元素,没有则返回nil
BLPOP和BRPOP与LPOP和RPOP类似,只不过在没有元素时等待指定时间而不是直接返回nil
LRANGE key star end返回一段表范围内的所有元素【0下标开始计算】

Set类型

Redis的Set结构与Java中的HashSet类似,可以看作是一个value为null的HashMap。

  • 无序
  • 不可重复
  • 查找快
  • 支持交并补查询

Set常用语法

语法含义
SADD key member向set中添加一个或多个元素
SREM key element移除set中的指定元素
SCARD key返回set中元素的个数
SISMEMBER key member判断一个元素是否存在于set中
SMEMBERS获取set中所有元素
SINTER key1 key2key1 和 key2 交集
SDIFF key1 key2key1 和 key2 差集集

SortedSet类型

Redis的SortedSet是一个可排序的Set集合。与Java中的TreeSet类似,但底层数据结构差异很大。SortedSet中的每个元素都带有score属性,可以基于score属性对元素排序,底层是一个调表(SkipList)+Hash表

  • 可排序
  • 不可重复
  • 查询快

因为SortedSet可排序特性,经常用来实现排行榜这样的功能

SortedSet常用语法

语法含义
ZADD key score member添加一个或多个元素到SortedSet,如果已经存在则更新其score值
ZREM key member删除SortedSet中指定元素的score值
ZSCORE key member获取SortedSet中指定元素的score值
ZRANK key member获取SortedSet中指定元素排名【升序】
ZREVRANK key member获取SortedSet中指定元素排名【降序】
ZCOUNT key min max统计score值在给定范围内的所有元素的个数
ZINCRBY key increment member让SortedSet中指定元素自增,步长为指定的increment值
ZRANGE key min max按照score排序后,获取指定排名范围内的元素
ZRANGEBYSCORE key min max按照score排序后,获取指定score范围内的元素
ZINTER,ZUNION,ZDIFF交并差

Redis的Java客户端

Jedis快速入门

在这里插入图片描述

  • Jedis和Lettuce:这两个主要是提供了Redis命令对应的API,方便操作Redis。而SpringDataRedis又针对这两种做了抽象和封装
  • Jedis:语法和Redis类似,优点是使用快捷缺点是多线程环境下会出现不安全
  • Lettuce:依靠opsForXxx进行操作Redis数据库,可解决多线程不安全情况
  • Redisson:是在Redis基础上实现了分布式的可伸缩的Java数据结构。例如Map、Queue等。而且支持跨进程的同步机制:Lock、Semaphore等待,比较适合用来实现特殊功能需求

创建一个Maven项目,引入需要的依赖
Jedis官网

<!-- redis依赖 -->
<dependency>
    <groupId>redis.clients</groupId>
    <artifactId>jedis</artifactId>
    <version>4.3.1</version>
</dependency>
<!-- 测试依赖 -->
<dependency>
  <groupId>org.junit.jupiter</groupId>
  <artifactId>junit-jupiter</artifactId>
  <version>5.9.1</version>
  <scope>test</scope>
</dependency>

一个redis小测试

import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import redis.clients.jedis.Jedis;

import java.util.Map;

public class JedisTest {
    private Jedis jedis;

    @BeforeEach
    void setUp() {
        // 1.建立连接
        jedis = new Jedis("127.0.0.1", 6379);
        // jedis = JedisConnectionFactory.getJedis();
        // 2.设置密码
        jedis.auth("123456");
        // 3.选择数据库
        jedis.select(0);
    }

    @AfterEach
    void close() {
        if (jedis != null) {
            jedis.close();
        }
    }

    @Test
    void testString() {
        // 存数据
        String result = jedis.set("name", "张三");
        System.out.println("result = " + result);
        // 取数据
        String name = jedis.get("name");
        System.out.println("name = " + name);
    }

    @Test
    void testHash() {
        jedis.hset("user:1", "name", "张三");
        jedis.hset("user:1", "age", "13");
        jedis.hset("user:1", "sex", "male");

        Map<String, String> map = jedis.hgetAll("user:1");
        System.out.println(map);
    }
}

打开客户端可以看到已经成功插入String和Hash类型的数据
在这里插入图片描述
在这里插入图片描述

由于经常的断开连接,建立连接会有消耗。所以以创建一个连接池

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

public class JedisConnectionFactory {
    private static final JedisPool jedisPool;

    static {
        // 配置连接池
        JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
        // 最大连接数
        jedisPoolConfig.setMaxTotal(10);
        // 最大空闲连接
        jedisPoolConfig.setMaxIdle(10);
        // 最小空闲连接
        jedisPoolConfig.setMinIdle(0);
        // 等待空闲时间[ms]
        jedisPoolConfig.setMaxWaitMillis(100);
        // 创建连接池对象,参数:连接池配置,服务端IP,服务端接口,超时时间,密码
        jedisPool = new JedisPool(jedisPoolConfig, "127.0.0.1", 6379, 100, "123456");
    }

    public static Jedis getJedis() {
        return jedisPool.getResource();
    }
}

SpringDataRedis客户端

SpringDataRedis官网简介
可以看到Redis的支持
在这里插入图片描述

创建一个Spring项目,添加如下依赖
在这里插入图片描述
yml配置如下

spring:
  redis:
    host: 127.0.0.1
    port: 6379
    password: Cxf@19307193096
    lettuce:
      pool:
        max-active: 8 #最大连接数
        max-idle: 8 #最大空闲连接
        min-idle: 0 #最小空闲连接
        max-wait: 1000ms #超时时间

测试代码如下

package app;

import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.redis.core.RedisTemplate;

@SpringBootTest
class TestRedisTemplate {
    private RedisTemplate redisTemplate;

    @Autowired
    public TestRedisTemplate(RedisTemplate redisTemplate) {
        this.redisTemplate = redisTemplate;
    }

    @Test
    public void testString() {
        // 写入一条 String 数据
        redisTemplate.opsForValue().set("name", "张三");
        // 获取一条 String 数据
        Object name = redisTemplate.opsForValue().get("name");
        System.out.println(name);
    }
}

在这里插入图片描述

会发现已经是乱码,可读性很差,因此需要用到Redis的序列化。那么问题出现在哪儿呢?我们顺着RedisTemplate 部分源码阅读一下
在这里插入图片描述
主要是 key和value 的序列化。redis中key一般用的都是字符串类型,因此使用的是String类型的序列化
程序会先通过 afterPropertiesSet 确定序列化方式
在这里插入图片描述
查看默认的 defaultSerializer 的属性如下所示,是一个 null 。所以会使用默认的 JDK序列化工具
在这里插入图片描述
我们再看 this.defaultSerializer = new JdkSerializationRedisSerializer(this.classLoader != null ? this.classLoader : this.getClass().getClassLoader()); 方法
在这里插入图片描述
再看 (new SerializingConverter() 代码
在这里插入图片描述
再看 new DefaultSerializer() 代码
在这里插入图片描述
再看 serialize() 用的是 ObjectOutPutStream 序列化
在这里插入图片描述
上面了解了 JDK的序列化方式,SpringDataRedis集成了众多序列化工具,默认使用的是JDK序列化方式,对于普通对象而言使用则会出现一定乱码问题,SpringDataRedis更推荐使用大名鼎鼎的 Jackson 进行对对象序列化
在这里插入图片描述

自定义 Redis 的序列化器

package app.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.RedisSerializer;

@Configuration
public class RedisConfig {
    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {
        // 1.创建 RedisTemplate 对象
        RedisTemplate redisTemplate = new RedisTemplate();
        // 2.设置连接工厂
        redisTemplate.setConnectionFactory(redisConnectionFactory);
        // 创建 json 序列化工具
        GenericJackson2JsonRedisSerializer jsonRedisSerializer = new GenericJackson2JsonRedisSerializer();
        // 3.设置 key 序列化
        redisTemplate.setKeySerializer(RedisSerializer.string());
        redisTemplate.setHashKeySerializer(RedisSerializer.string());
        // 4.设置 value 序列化
        redisTemplate.setValueSerializer(jsonRedisSerializer);
        redisTemplate.setHashValueSerializer(jsonRedisSerializer);
        // 5.返回 RedisTemplate
        return redisTemplate;
    }
}

在这里插入图片描述
测试结果如下所示
在这里插入图片描述
我们再测试一下对象的存储结果
在这里插入图片描述
再去redis数据库中查看
在这里插入图片描述

在这里插入图片描述
说明:对于普通字符串 “张三” 直接按照String类型存入到了redis中;而对于 User 对象则被 Jackson 序列化为了为了 json 类型的数据,为了能够方便通过 json 数据返回序列化出 User 对象还会多存入一条属性 "@class": "app.pojo.User"。然而这样虽然反序列化方便了,但是数据量堆叠起来之后会给redis带了额外的内存开销

StringRedisTemplate 使用String序列化器

因此为了节省内存,一般并不会使用JSON序列化器,而是统一使用String序列化器,要求之存储String类型的key和value。当需要的时候在手动序列化或反序列化。
在这里插入图片描述
主要利用jackson的 ObjectMapper 类来实现手动的序列化和反序列化而不是通过Redis自带的JSON序列化工具

读写String

package app;

import app.pojo.User;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.redis.core.StringRedisTemplate;

@SpringBootTest
public class TestStringRedisTemplate {
    private StringRedisTemplate stringRedisTemplate;
    private static final ObjectMapper mapper = new ObjectMapper();

    @Autowired
    public TestStringRedisTemplate(StringRedisTemplate stringRedisTemplate) {
        this.stringRedisTemplate = stringRedisTemplate;
    }

    @Test
    public void testString() {
        // 写入 String 数据
        stringRedisTemplate.opsForValue().set("name", "张三");
        // 读取 String 数据
        String name = stringRedisTemplate.opsForValue().get("name");
        System.out.println(name);
    }

    @Test
    public void testSaveUser() throws JsonProcessingException {
        // 创建对象
        User user = new User("李四", 24);
        // 手动序列化
        String json = mapper.writeValueAsString(user);
        // 写入 User 数据
        stringRedisTemplate.opsForValue().set("user", json);
        // 读取 User 数据
        String jsonUser = stringRedisTemplate.opsForValue().get("user");
        System.out.println("redis读取结果: " + jsonUser);
        // 手动反序列化
        user = mapper.readValue(jsonUser, User.class);
        System.out.println("jsonUser反序列化: " + user);
    }
}

会发现Redis在存储的时候已经消除掉多余的数据
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

读写Hash

在处理 Hash 类型的时候,语法hset有些不同,更偏向于 Java 语法 put

@Test
public void testSaveHash(){
    stringRedisTemplate.opsForHash().put("user:1", "name", "张三");
    stringRedisTemplate.opsForHash().put("user:1", "age", "23");
    // 获取单个字段
    String name = (String) stringRedisTemplate.opsForHash().get("user:1", "name");
    System.out.println(name);
    // 获取全部
    Map<Object, Object> entries = stringRedisTemplate.opsForHash().entries("user:1");
    System.out.println(entries);
}

在这里插入图片描述
在这里插入图片描述

总结
方案一

  1. 自定义 RedisTemplate
  2. 修改 RedisTemplate 序列化器为 GenericJackson2JsonRedisSerializer

方案二

  1. 使用 StringRedisTemplate
  2. json序列化处理之后再写入redis
  3. 读取完redis之后再json反序列化成对象

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/148116.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

jenkins前端页面自动运维值yarn编译运行项目

配置步骤如下 首先需要在系统管理中心安装node相关插件 安装完成之后&#xff0c;在系统管理——>全局工具配置——>NodeJS 点击新增Nodejs 此处自定义别名&#xff0c;我这里是Nodejs16&#xff0c;取消自动安装前面的复选框&#xff0c;下方选择我们的nodejs安装目录&…

云服务器定时执行python脚本

文章目录前言crontab简介基本语法定时任务具体内容python 脚本定时任务前言 在服务器上定时执行任务有两种方式&#xff0c;一种是at定时任务&#xff0c;一种是crond任务调度&#xff0c;at命令是一次性定时计划任务&#xff0c;at的守护进程 atd会以后台模式运行&#xff0c…

Spring注解之@validated使用

概念 spring-boot中可以用validated来校验数据&#xff0c;如果数据异常则会统一抛出异常&#xff0c;方便异常中心统一处理。 注解源码&#xff1a; Validated 作用在类、方法和参数上 Target({ElementType.TYPE, ElementType.METHOD, ElementType.PARAMETER}) Retention(…

python初级教程十一 urllib

urllib Python urllib 库用于操作网页 URL&#xff0c;并对网页的内容进行抓取处理。 本文主要介绍 Python3 的 urllib。 urllib 包 包含以下几个模块&#xff1a; urllib.request - 打开和读取 URL。 urllib.error - 包含 urllib.request 抛出的异常。 urllib.parse - 解…

【Python入门指北】操作数据库

文章目录一、1.数据库2.练手案例二、redis数据库一、 1.创建一个数据库 [guanmaster1 ~]$ mysql -uroot -p123456 mysql: [Warning] Using a password on the command line interface can be insecure. Welcome to the MySQL monitor. Commands end with ; or \g. Your MySQL…

Apache IoTDB PMC 乔嘉林荣获 2022 杰出开源贡献者|开源技术强大,开源文化活跃...

2022 年 12 月 29 日至 30 日&#xff0c;2022 木兰峰会正式召开&#xff0c;会上发布了中国开源云联盟 2022 年度评选名单。本次评审专家包括数十位开源领域专家、社区领袖、科研院所专家&#xff0c;共评选出杰出开源贡献者 3 人。其中&#xff0c;清华大学助理研究员、博士后…

【QT开发笔记-基础篇】| 第五章 绘图QPainter | 5.16 完结和后续:《Qt开发专题-自定义控件》

本节对应的视频讲解&#xff1a;B_站_视_频 https://www.bilibili.com/video/BV1NW4y1K7eL 1. 为什么需要自定义控件 绘图最大的一个应用场景就是自定义控件&#xff0c;Qt 本身提供的一些控件是有限的&#xff0c;并且它提供的一些控件很可能不满足我们的需要 这种情况下&a…

pygame - 图片移动优化

目录 一、优化过程分析 1、pygame - 图片移动中图片移动模式 2、优化过程 二、代码段 1、附注释 2、无注释 三、效果展示 一、优化过程分析 1、pygame - 图片移动中图片移动模式 按一次方向键&#xff0c;图片移动一小步&#xff1b; 若需要一直往某个方向移动&…

【LaTex】LaTex 极简安装教程

文章目录Latex 安装教程1. 下载texlive.iso2. 点击装载3. 运行 install-tl-windows.bat 进行安装4. 验证是否安装成功5. 安装开发工具Latex 安装教程 1. 下载texlive.iso https://mirrors.tuna.tsinghua.edu.cn/ctan/systems/texlive/Images/ 2. 点击装载 下载完成后&#x…

【博客580】内核如何决定数据包的源ip

内核如何决定数据包的源ip 1、Traversing of packets Receive&#xff1a; 某个interface收到数据包 -> PREROUTING (manage, nat) -> routing -> 是发送给本机的数据包? -> INPUT (manage, filter) -> app 不是 -> FORWARD (manage, filter) -> POST…

《Dubbo源码剖析与实战》学习笔记 Day1

流量控制 为了系统的健壮性和稳定性考虑&#xff0c;通常要对访问进行有效的控制&#xff0c;防止流量突然暴增&#xff0c;引发一系列服务雪崩效应。 1.单机限流 大概逻辑就是把每个请求当作一个小朋友&#xff0c;把最细粒度Dubbo服务中的方法当作游乐场&#xff0c;每个方…

Go第 12 章 :1-家庭收支记账软件项目

Go第 12 章 &#xff1a;1-家庭收支记账软件项目 12.1 项目开发流程说明 12.2 项目需求说明 模拟实现基于文本界面的《家庭记账软件》该软件能够记录家庭的收入、支出&#xff0c;并能够打印收支明细表 12.3 项目的界面 12.4 项目代码实现 12.4.1 实现基本功能(先使用面向过…

Windows 10/11 中的快速录屏的 5 种方法

在当今世界&#xff0c;出于各种原因&#xff0c;人们更喜欢录制他们的 PC 屏幕。录制计算机屏幕的功能对于在线学习、录制在线会议的特定亮点或帮助您制作抖音视频至关重要。不管是什么原因&#xff0c;屏幕录制让许多人的生活更轻松。截屏是有效的&#xff0c;但录制有助于您…

聚簇索引,二级索引,MRR,联合索引和自适应哈希索引-详细总结

目录 聚簇索引&#xff0c;二级索引&#xff0c;MRR&#xff0c;联合索引和自适应哈希索引-详细总结 聚簇索引 定义&#xff1a; 问题&#xff1a;为什么不采用B树作为MySQL表数据底层的存储数据结构&#xff1f; 作图&#xff1a; 问题&#xff1a;如果数据库表没有设置…

base64和base64url编解码

文章目录[toc]前言1. Base64编码原理2. 加解密图示3. base64编码Code3. base64url编码Code前言 一个字节可以表示256种数值&#xff0c;但是由于一些字节在网络中有特殊的含义。所以当传输字节内容时就不能传输这些具有控制功能的字符。具体的做法就是将这些字符进行转码。   …

A. Sasha and a Bit of Relax(异或的性质 + 前缀和的性质)

Problem - 1109A - Codeforces Sasha喜欢编程。有一次&#xff0c;在一场很长时间的比赛中&#xff0c;萨沙觉得他有点累了&#xff0c;需要放松一下。他照做了。但由于萨沙不是一个普通人&#xff0c;他更喜欢不同寻常地放松。在闲暇时间&#xff0c;萨沙喜欢解决未解决的问题…

JAVA本科毕业设计家庭财务管理系统源码+数据库,基于springboot + mybatis + mysql5.7

家庭财务管理系统 下载地址&#xff1a;JAVA本科毕业设计家庭财务管理系统源码数据库 介绍 1.cwgl 财务管理系统 提供sql 后面系统在另一个版本修改 此版本功能简单 可能有很多没有完善的地方 仅供参考 2.ffms&#xff08;Family Financial Management System&#xff09; …

linux 编译 c++ 静态库(包含类的实例化)给 c 程序调用

文章目录使用场景代码示例编译运行示例程序参考文章使用场景 c 是面向对象的编程语言&#xff0c;比较方便实现某些第三方库&#xff0c;比如翻译其他面向对象语言的代码&#xff0c;比 c 语言要方便的多。而 c 语言跟 c 很亲和&#xff0c;可以用 c 来实现&#xff0c;编译成…

SAP灵活工作流客制化值帮助

目录 1. 创建值帮助CDS视图 2. 添加值帮助CDS视图对应的ODATA服务 3. 在灵活工作流中设置搜索帮助 4. 效果展示 1. 创建值帮助CDS视图 创建值帮助数据源视图 创建值帮助视图 Dummy表结构如下 2. 注册值帮助CDS视图对应的ODATA服务 事务代码 /N/IWFND/MAINT_SERVICE 选择添加…

【mysql】-【innodb数据存储结构】

文章目录数据库的存储结构&#xff1a;页磁盘与内存交互基本单位&#xff1a;页页结构概述页的大小页的上层结构页的内部结构数据库的存储结构&#xff1a;页 一、索引结构给我们提供了高效的索索隐方式&#xff0c;不过索引信息以及数据记录都是保存在文件上的&#xff0c;确…