【Redis】数据类型详解及其应用场景

目录

    • Redis 常⻅数据类型
      • 预备知识
        • 基本全局命令
          • 小结
        • 数据结构和内部编码
        • 单线程架构
          • 引出单线程模型
          • 为什么单线程还能这么快


Redis 常⻅数据类型

Redis 提供了 5 种数据结构,理解每种数据结构的特点对于 Redis 开发运维⾮常重要,同时掌握每种数据结构的常⻅命令,会在使⽤ Redis 的时候做到游刃有余。本章内容如下:

  • 预备知识:⼏个全局(generic)命令,数据结构和内部编码,单线程模式机制分析。
  • 5 种数据结构的特点、命令使⽤、应⽤场景⽰例。
  • 键遍历、数据库管理。

预备知识

在正式介绍 5 种数据结构之前,了解⼀下 Redis 的⼀些全局命令、数据结构和内部编码、单线程命令处理机制是⼗分必要的,它们能为后⾯内容的学习打下⼀个良好的基础.主要体现在两个⽅⾯:

  1. Redis 的命令有上百个,如果纯靠死记硬背⽐较困难,但是如果理解 Redis 的⼀些机制,会发现这些命令有很强的通⽤性。
  2. Redis 不是万⾦油,有些数据结构和命令必须在特定场景下使⽤,⼀旦使⽤不当可能对 Redis 本⾝或者应⽤本⾝造成致命伤害。
基本全局命令

Redis 有 5 种数据结构,但它们都是键值对种的值,对于键来说有⼀些通⽤的命令。

必须要进入 redis-cli 客户端程序,才能输入 redis 命令
全局命令就是能够搭配任意一个数据结构使用的命令

KEYS

keys

查询当前服务器上匹配的 key,返回所有满⾜样式(pattern)的 key。⽀持如下统配样式。
通过一些特殊符号(通配符)来描述的模样,匹配上述模样的 key 就能被查询出来

pattern 包含特殊符号的字符串,是去描述另外的字符串是什么样的

  • h?llo 匹配 hello , hallohxllo
  • h*llo 匹配 hlloheeeello
  • h[ae]llo 匹配 hellohallo 但不匹配 hillo
  • h[^e]llo 匹配 hallo , hbllo , … 但不匹配 hello
  • h[a-b]llo 匹配 hallohbllo

?匹配任意一个字符

  • 匹配任意零个或者多个字符
    [abcde]只能匹配 abcde 的任意一个,相当于给出固定选项
    [^e]排除 e,只有 e 匹配不了,其他都能匹配
    [a-e]匹配 a 到 e 之间的字符,包括 a 和 e

语法:

KEYS pattern

命令有效版本:1.0.0 之后

时间复杂度:O(N)

所以在生产环境上,一般禁止使用 keys 命令,尤其是像 keys _ 这种,这是查询 Redis 存储所有的 key。Redis 经常被用于做缓存,挡在 MySQL 前面,如果 Redis 被 keys _ 这样一个命令阻塞住了,其他查询 Redis 操作间超时了,此时这些请求就会直接查数据库,如果请求过多 MySQL 就容易挂了。

  1. 办公环境:办公的电脑
  2. 开发环境:有时就是和办公环境一样的(前端/客户端),有时开发环境就是一个单独的服务器(后端,因为有些程序比较复杂)
    1. 编译一次时间比较久,比如 C++,就需要一台高性能的服务器
    2. 有些程序一启动就消耗很多 CPU 和内存资源
    3. 有的程序比较依赖 Linux,在 Windows 环境搭不起来
  1. 测试环境:一般是比较好的服务器,测试工程师用的
  2. 生产环境(线上环境):与前面三个环境相对的,前面三个是线下环境,线上环境是外界用户访问的,线下环境是外界用户无法访问的,这是影响公司营收的

返回值:匹配 pattern 的所有 key。

⽰例:

redis> MSET firstname Jack lastname Stuntman age 35
"OK"
redis> KEYS *name*
1) "firstname"
2) "lastname"
redis> KEYS a??
1) "age"
redis> KEYS *
1) "age"
2) "firstname"
3) "lastname"

EXISTS

exists

判断某个 key 是否存在。

语法:

EXISTS key [key ...]

命令有效版本:1.0.0 之后

时间复杂度:O(1),严谨点应该是多少个 key 就多少个 O(1)

Redis 组织这些 key 就是按照哈希表的方式来组织的。

Redis 支持很多数据结构指的是一个 value 可以是一些复杂的数据结构。Redis 自身的这些键值对是通过哈希表组织的,具体的某个值又可以是一些数据结构。

返回值:key 存在的个数。

⽰例:

 redis> SET key1 "Hello""OK"redis> EXISTS key1(integer) 1redis> EXISTS nosuchkey(integer) 0redis> SET key2 "World""OK"redis> EXISTS key1 key2 nosuchkey(integer) 2

DEL

del

删除指定的 key。

语法:

DEL key [key ...]

命令有效版本:1.0.0 之后

时间复杂度:O(1)

返回值:删除掉的 key 的个数。

⽰例:

 redis> SET key1 "Hello""OK"redis> SET key2 "World""OK"redis> DEL key1 key2 key3(integer) 2

在 MySQL ,像删除一类的操作,都是比较危险的,删除后数据就没有了。但在 Redis 主要的应用场主要是作为缓存,此时存的是一个热点数据,全量数据在 MySQL 之类的数据库中,此时直接把 Redis 的 key 删除了几个,一般来说问题不大,除非是删除了很多,这样就会导致请求直接发送到 MySQL,就容易把 MySQL 搞挂。

如果把 Redis 作为数据库存储,这样误删数据就问题大了。

如果把 Redis 作为消息队列,影响大不大就具体情况具体分析了。

EXPIRE

expire

为指定的 key 添加秒级的过期时间(Time To Live TTL),单位是秒,超过时间 key 就被自动删除了,比如手机验证码,基于 Redis 实现分布式锁(为避免不能正确解锁的情况,通常都会在加锁的时候设置过期时间,Redis 实现分布式锁就是给 Redis 里写一个特殊的 key value)。

pexpire 单位是毫秒,其他和 expire 是一样的

语法:

EXPIRE key seconds

key 必须存在

命令有效版本:1.0.0 之后

时间复杂度:O(1)

返回值:1 表⽰设置成功。0 表⽰设置失败。

⽰例:

TTL

ttl

time to live

获取指定 key 的过期时间,秒级。

pttl 单位是毫秒,其他和 ttl 是相似的

语法:

TTL key

命令有效版本:1.0.0 之后

时间复杂度:O(1)

返回值:剩余过期时间。-1 表⽰没有关联过期时间,-2 表⽰ key 不存在。

⽰例:

 redis> SET mykey "Hello""OK"redis> EXPIRE mykey 10(integer) 1redis> TTL mykey(integer) 10

EXPIRE 和 TTL 命令都有对应的⽀持毫秒为单位的版本:PEXPIRE 和 PTTL,详细⽤法就不再介绍了

一个 Redis 中可能同时存在很多 key,很大一部分可能都有过期时间,Redis 服务器是怎么知道哪些 key 已经过期要被删除,哪些 key 还没过期?

如果直接遍历所有的 key,肯定是效率很低的。

Redis 整体的策略:

  1. 定期删除:这个定期删除也采用了一个抽取样本的方式,每次只抽取一部分,保证这个抽取检查的过程足够快,不要浪费太多时间
  2. 惰性删除:过期了不会立即删除,但一旦访问到了 Redis 就立刻出发删除 key 的操作,同时返回一个 nil

以上策略的原因:因为 Redis 是单线程的程序,主要的任务是处理每个命令,如果扫描过期 key 太耗费时间就导致正常处理请求命令被阻塞了

以上策略的效果是一般的,仍然可能有很多过期的 key 被残留。对以上 Redis 也是有补充的,比如一系列的内存淘汰策略

注意:

  1. Redis 中没有采取定时器的方式来实现过期 key 的删除
  2. 如果有多个 key 过期,也可以通过一个定时器(基于优先级队列或者时间轮)来高效/节省 CPU 的前提下来处理多个 key

作者没有采用定时器的原因?可能是因为如果基于定时器实现就要引入多线程了,而 Redis 早期版本就是奠定了单线程的基调,多线程就不符合作者初衷

定时器的实现方式(Redis 并没有这个做):

  1. 优先级队列:
    1. 假定很多 key 都设置了过期时间,放入优先级队列后,指定优先级规则是过期时间早的先出队列,此时扫描线程只需要盯着队首元素即可
    2. 但这个是扫描线程也不需要检查得太频繁,可以根据当前时刻和队首元素的过期时间设置一个等待
    3. 此时如果有新任务添加进来,就唤醒一下刚才的线程,重新检查一下队首元素,再根据时间差距重新调整阻塞时间即可
  1. 时间轮:
    1. 把时间划分成很多的格子,划分的粒度看实际需求
    2. 每个格子都挂着一个链表,每个链表都代表一个要执行的任务
    3. 此时设置一个指针遍历这个轮,每隔固定的时间走一个格子
    4. 此时添加一个 key 时,把其过期时间除以指针移动的固定时间,放到对应的格子
    5. 对于时间轮多少个格子以及移动的固定时间都是根据需求设置的

Redis 源码中,一个比较核心的机制是事件循环

关于键过期机制,可以参考图 2-1 所⽰。

图 2-1 键的过期机制

TYPE

返回 key 对应的数据类型。

Redis 所有的 key 都是 string,但 key 对应的 value 可能有多种类型

语法:

TYPE key

命令有效版本:1.0.0 之后

时间复杂度:O(1)

返回值: none , string , list , set , zset , hashstream(这个是 Redis 作为消息队列使用的 value)。

⽰例:

 redis> SET key1 "value""OK"redis> LPUSH key2 "value"(integer) 1redis> SADD key3 "value"(integer) 1redis> TYPE key1"string"redis> TYPE key2"list"redis> TYPE key3"set"
小结
  • keys:用来查看匹配规则的 key
  • exists:用来判定指定 key 是否存在
  • del:删除指定的 key
  • expire:给 key 设置过期时间
  • ttl:查询 key 的过期时间
  • type:查询 key 对应的 value 的类型

以上只是给出⼏个通⽤的命令,为 5 种数据结构的使⽤(实际上 Redis 支持10个数据类型)做⼀个热⾝,后续将对键管理做⼀个更为详细的介绍。

数据结构和内部编码

type 命令实际返回的就是当前键的数据结构类型,它们分别是:string(字符串)、list(列表)、hash(哈希)、set(集合)、zset(有序集合),但这些只是 Redis 对外的数据结构,如图 2-2 所⽰。

图 2-2 Redis 的 5 种数据类型

这5种数据结构相当于 Java 中的:

  1. String
  2. HashMap
  3. List
  4. Set
  5. 这个找不太到,除了存储 member 外还需要存储一个 score(权重)

实际上 Redis 针对每种数据结构都有⾃⼰的底层内部编码实现,⽽且是多种实现,这样 Redis 会在合适的场景选择合适的内部编码,如表 2-1 所⽰。

表 2-1 Redis 数据结构和内部编码

数据类型内部编码说明
stringraw最基本的字符串
stringint当 value 是一个整数时,Redis 很可能直接用 int 保存
stringembstr针对短字符串进行的特殊优化
hashhashtable最基本的哈希表,但这并不是 Java 标准库的那个 HashTable
hashziplist压缩列表,针对哈希表里面元素较少的情况,能够节省空间
listlinkedlist链表
listziplist压缩列表
sethashtable
setintset针对集合中存的都是整数
zsetskiplist跳表,也是链表,但每个节点上有多个指针域
zsetziplist

可以看到每种数据结构都有⾄少两种以上的内部编码实现,Redis 会自动根据当前的实际情况选择内部的编码方式,自动适应的,例如 list 数据结构包含了 linkedlist 和 ziplist 两种内部编码。同时有些内部编码,例如 ziplist,可以作为多种数据结构的内部实现,可以通过 object encoding 命令查询内部编码:

127.0.0.1:6379> set hello world
OK
127.0.0.1:6379> lpush mylist a b c
(integer) 3
127.0.0.1:6379> object encoding hello
"embstr"
127.0.0.1:6379> object encoding mylist
"quicklist"

可以看到 hello 对应值的内部编码是 embstr,键 mylist 对应值的内部编码是 ziplist。

从 Redis 3.2 开始,对于 list 引入了新的实现方式 quicklist,就直接代替了 linkedlist 和 ziplist,同时兼顾了两个的优点,quicklist 就是一个链表,每个元素又是一个 ziplist,把空间和效率都折中兼顾到。

注意,只是对于 list 这个数据类型是这样,别的还是按照上述表格

Redis 这样设计有两个好处:

  1. 可以改进内部编码,⽽对外的数据结构和命令没有任何影响,这样⼀旦开发出更优秀的内部编码,⽆需改动外部数据结构和命令,例如 Redis 3.2 提供了 quicklist,结合了 ziplist 和 linkedlist 两者的优势,为列表类型提供了⼀种更为优秀的内部编码实现,⽽对⽤⼾来说基本⽆感知。
  2. 多种内部编码实现可以在不同场景下发挥各⾃的优势,例如 ziplist ⽐较节省内存,但是在列表元素⽐较多的情况下,性能会下降,这时候 Redis 会根据配置选项将列表类型的内部实现转换为linkedlist,整个过程⽤⼾同样⽆感知。
单线程架构

Redis 使⽤了单线程架构来实现⾼性能的内存数据库服务,本节⾸先通过多个客⼾端命令调⽤的例⼦说明 Redis 单线程命令处理机制,接着分析 Redis 单线程模型为什么性能如此之⾼,最终给出为什么理解单线程模型是使⽤和运维 Redis 的关键。

Redis 只使用一个线程处理所有的命令请求,而不是 Redis 服务器进程内部真的只有一个线程,有多个线程,多个线程是在处理网络IO

引出单线程模型

现在开启了三个 redis-cli 客⼾端同时执⾏命令。

客⼾端 1 设置⼀个字符串键值对:

客⼾端 2 对 counter 做⾃增操作:

客⼾端 3 对 counter 做⾃增操作:

我们已经知道从客⼾端发送的命令经历了:发送命令、执行命令、返回结果三个阶段,其中我们重点关注第 2 步。我们所谓的 Redis 是采⽤单线程模型执⾏命令的是指:虽然三个客户端看起来是同时要求 Redis 去执行命令的,但微观⻆度,这些命令还是采⽤线性⽅式去执⾏的,只是原则上命令的执⾏顺序是不确定的,但⼀定不会有两条命令被同步执⾏,如图 2-3、2-4、2-5 所⽰,可以想象 Redis 内部只有⼀个服务窗⼝,多个客⼾端按照它们达到的先后顺序被排队在窗⼝前,依次接受 Redis 的服务,所以两条 incr 命令⽆论执⾏顺序,结果⼀定是 2,不会发⽣并发问题,这个就是 Redis 的单线程执⾏模型

图 2-3 宏观上同时要求服务的客⼾端

图 2-4 微观上客⼾端发送命令的时间有先后次序的

图 2-5 Redis 的单线程模型

为什么单线程还能这么快

这是个重要的面试题

通常来讲,单线程处理能⼒要⽐多线程差,例如有 10000 公⽄货物,每辆⻋的运载能⼒是每次 200 公⽄,那么要 50 次才能完成;但是如果有 50 辆⻋,只要安排合理,只需要依次就可以完成任务。那么为什么 Redis 使⽤单线程模型会达到每秒万级别的处理能⼒呢?可以将其归结为三点:

  1. 纯内存访问。Redis 将所有数据放在内存中,内存的响应时⻓⼤约为 100 纳秒,这是 Redis 达到每秒万级别访问的重要基础。
  2. 核心功能更简单。Redis 核心功能比数据库的核心功能更简单,数据库对于数据的插入删除查询都有更复杂的功能支持,这样的功能势必要花费更多的开销。
  3. 非阻塞 IO。Redis 使⽤ epoll 作为 I/O 多路复⽤技术的实现,再加上 Redis ⾃⾝的事件处理模型将 epoll 中的连接、读写、关闭都转换为事件,不在⽹络 I/O 上浪费过多的时间,如图 2-6 所⽰。

一个线程就可以管理多个 socket,对于 TCP 来说,服务器每次要服务一个客户端就要安排一个 socket,一个服务器服务多个客户端,就需要多个 socket,但很多情况下每个客户端和服务器之间的通信也没那么频繁,此时很多 socket 大部分时间都是静默的,是没有数据需要传输的,同一时刻只有少数 socket 是活跃的,因此就用个 IO多路复用,一个线程来处理多个 socket。

这是操作系统给程序员提供的机制,提供了一套 API,内部的功能都是操作系统实现的。

Linux 提供的 IO多路复用 主要是三套 API:

  1. select
  2. poll
  3. epoll

epoll 是个事件通知/回调机制,一个线程可以同时做多件事情,能高效完成多件事,但前提是这多件事交互是互不影响的,大部分时间都在等。

C++ 可以直接使用 Linux 原生的 epoll API;Java 可以使用 NIO,这是标准库提供的一组类,底层就是封装了 epoll。

  1. 单线程避免了线程切换和竞态产生的消耗。单线程可以简化数据结构和算法的实现,让程序模型更简单;其次多线程避免了在线程竞争同⼀份共享数据时带来的切换和等待消耗。

图 2-6 Redis 使⽤ I/O 多路复⽤模型

虽然单线程给 Redis 带来很多好处,但还是有⼀个致命的问题:对于单个命令的执⾏时间都是有要求的。如果某个命令执⾏过⻓,会导致其他命令全部处于等待队列中,迟迟等不到响应,造成客⼾端的阻塞,对于 Redis 这种⾼性能的服务来说是⾮常严重的,所以 Redis 是面向快速执行场景的数据库

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Postman接口测试项目实战

第 1 章 什么是接口测试 1.1、为什么要进行接口测试 目前除了特别Low的公司外,开发都是前后端分离的,就是说前端有前端的工程师进行编码,后端有后端的工程师进行编码,前后端进行数据基本都是通过接口进行交互的。 1.2、接口测…

zookeeper源码分析之事务请求处理

一.参考 zookeeper启动和选举的源码分析参考之前的帖子. 二.源码 1.职责链模式. 每次经过的processor都是异步处理,加入当前processor的队列,然后新的线程从队列里面取出数据处理. PrepRequestProcessor 检查ACL权限,创建ChangeRecord. SyncRequest…

ArcGIS空间自相关Moran‘s I——探究人口空间格局的20年变迁

先了解什么是莫兰指数? 莫兰指数(Morans I)是一种用于衡量空间自相关性的统计量,即它可以帮助我们了解一个地理区域内的观测值是否彼此相关以及这种相关性的强度和方向。 莫兰指数分类: 全局莫兰指数 (Global Moran…

聊聊如何利用ingress-nginx实现应用层容灾

前言 容灾是一种主动的风险管理策略,旨在通过构建和维护异地的冗余系统,确保在面临灾难性事件时,关键业务能够持续运作,数据能够得到保护,从而最大限度地减少对组织运营的影响和潜在经济损失。因此容灾的重要性不言而…

zabbix实战-磁盘空间告警

1.创建监控项 选择&#xff1a;键值&#xff1a;vfs.fs.size[fs,<mode>] 。 直接写 vfs.fs.size[fs,<mode>]是不出数据的。我们要写具体的值 &#xff1a;vfs.fs.size[/,free] &#xff0c;这个表示查看根的剩余空间。 2.创建图形 为磁盘剩余空间监控项创建图形&am…

redis 遍渐进式历

1.scan cursor [match pattern] [coutn] [type]:以渐进式的方式进行建的遍历 cursor:是光标 指向当前遍历的位置 设置成0表示当前从0开始获取 math parttern &#xff1a;和keys命令一样的 keys * count: 限制一次遍历能够获取到多少个 元素默认是10 type :这次遍历只想获取…

[Python学习日记-10] Python中的流程控制(if...else...)

[Python学习日记-10] Python中的流程控制&#xff08;if...else...&#xff09; 简介 缩进 单分支 双分支 多分支 练习 简介 假如把写程序比做走路&#xff0c;那我们到现在为止&#xff0c;一直走的都是直路&#xff0c;还没遇到过分叉口&#xff0c;想象现实中&#x…

【python】Python实现XGBoost算法的详细理论讲解与应用实战

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

Python编码系列—Python数据可视化:Matplotlib与Seaborn的实战应用

&#x1f31f;&#x1f31f; 欢迎来到我的技术小筑&#xff0c;一个专为技术探索者打造的交流空间。在这里&#xff0c;我们不仅分享代码的智慧&#xff0c;还探讨技术的深度与广度。无论您是资深开发者还是技术新手&#xff0c;这里都有一片属于您的天空。让我们在知识的海洋中…

Ps:首选项 - 单位与标尺

Ps菜单&#xff1a;编辑/首选项 Edit/Preferences 快捷键&#xff1a;Ctrl K Photoshop 首选项中的“单位与标尺” Units & Rulers选项卡允许用户根据工作需求定制 Photoshop 的测量单位和标尺显示方式。这对于保持工作的一致性和精确性&#xff0c;尤其是在跨设备或跨平台…

专题--自底向上的计算机网络(物理层)

目录 计算机网络概述 物理层 数据链路层 网络层 运输层 应用层 网络安全 详细见http://t.csdnimg.cn/MY5aI http://t.csdnimg.cn/8Ipa4 http://t.csdnimg.cn/uvMxS 信道复用技术不仅在物理层有运用&#xff0c;在数据链路层也确实需要信道复用技术。‌ 数据链路层是…

第N8周:使用Word2vec实现文本分类

本文为365天深度学习训练营 中的学习记录博客原作者&#xff1a;K同学啊 一、数据预处理 任务说明: 本次将加入Word2vec使用PyTorch实现中文文本分类&#xff0c;Word2Vec 则是其中的一种词嵌入方法&#xff0c;是一种用于生成词向量的浅层神经网络模型&#xff0c;由Tomas M…

隐语隐私计算实训营「数据分析」第 5 课:隐语PSI介绍及开发实践

深入理解隐语(SecretFlow)中的PSI技术 隐私集合求交(Private Set Intersection, PSI)是隐私计算中的一个重要技术,它允许多方在不泄露自己数据的前提下找出共同的数据交集。在本文中,我们将深入探讨SecretFlow(隐语)中PSI的实现和应用。 PSI的基本概念 PSI是一种特殊的安全多…

鸿蒙内核源码分析(时钟任务篇)

时钟概念 时间是非常重要的概念&#xff0c;我们整个学生阶段有个东西很重要,就是校园铃声. 它控制着上课,下课,吃饭,睡觉的节奏.没有它学校的管理就乱套了,老师拖课想拖多久就多久,那可不行,下课铃声一响就是在告诉老师时间到了,该停止了让学生HAPPY去了. 操作系统也一样&…

php源码编译与初始化

1 php源码编译 解压 yum install -y bzip2 # 安装解压工具 tar -xf php-7.4.12.tar.bz2 # 解压文件./condigure ./configure --prefix/usr/local/php --with-config-file-path/usr/local/php/etc --enable-fpm --with-fpm-usernginx --with-fpm-groupnginx --with-curl --wi…

线程面试题

1.JDK自带的线程池有哪些&#xff1f; 2.线程池中核心线程数与最大线程数与缓冲任务队列的关系&#xff1f; 先使用核心线程执行任务。 当核心线程不足时&#xff0c;新任务入队列等待。 当队列满且线程数未达最大值时&#xff0c;增加非核心线程执行任务。 当队列满且线程…

Leetcode每日刷题之209.长度最小的子数组(C++)

1.题目解析 根据题目我们知道所给的数组均是正整数&#xff0c;我们需要找到的是该数组的子数组&#xff0c;使其子数组内所有元素之和大于或等于给出的目标数字target&#xff0c;然后返回其长度&#xff0c;最终找出所以满足条件的子数组&#xff0c;并且要返回长度最小的子数…

网络硬盘录像机NVR程序源码海思3520D NVR 安防监控智能升级运用方案

随着安防技术的不断发展&#xff0c;传统的监控系统正逐步向智能化方向转变。海思Hi3520D作为一款高性能的网络视频处理芯片&#xff0c;在NVR&#xff08;网络视频录像机&#xff09;领域有着广泛的应用。本方案旨在探讨如何利用海思Hi3520D芯片的强大功能对现有的NVR系统进行…

vue2使用天地图

需求&#xff1a;用vue2使用天地图展示对应点位数据以及开发中出现的问题等&#xff0c;其实天地图的写法和百度地图差不多 注意&#xff01;&#xff01;&#xff01;天地图的接口不稳定&#xff0c;时常报错418&#xff0c;官网也是一样的情况&#xff0c;推荐还是使用百度或…

C++:C/C++的内存管理

目录 C/C内存分布 C语言中动态内存管理方式 C内存管理方式 new/delete操作内置类型 new/delete操作自定义类型 operator new与operator delete函数 new和delete的实现原理 定位new表达式 常见问题 malloc/free和new/delete的区别 内存泄漏 C/C内存分布 我们先来看以…