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

【征服redis1】基础数据类型详解和应用案例

博客计划 ,我们从redis开始,主要是因为这一块内容的重要性不亚于数据库,但是很多人往往对redis的问题感到陌生,所以我们先来研究一下。

本篇,我们先看一下redis的基础数据类型详解和应用案例。

1.redis概述

以mysql为代表的数据库,相信很多人都非常熟悉,这是数据持久化存储的最佳方式。但是在更新频繁或者访问强度高的场景下,数据库的性能捉襟见肘,这时候我们使用Redis作为缓存更加合适。缓存是redis最主要的功能,此外,redis还在分布式锁、异步消息队列等方面有重要应用。 

Redis 是一个开源的在内存中存储数据的结构化键值数据库,它支持多种类型的数据结构,如字符串、列表、集合、哈希、有序集合等。此外,Redis还支持事务、持久化、LUA脚本、LRU清除、自动故障转移等功能。

Redis特点

  1. 性能高效:Redis能读的速度是110000次/s,写的速度是81000次/s,因此被广泛应用于缓存场景。
  2. 支持多种数据类型:Redis支持最常见的数据类型,为复杂的应用提供了广泛的可能性。
  3. 数据持久化:Redis可以将数据保存在磁盘上,重启后可以再次加载进行使用。
  4. 支持消息队列:Redis还可以用作消息队列使用,它提供了发布/订阅等相关功能。
  5. 支持事务:Redis支持事务,可以一次执行多个命令,中间不会被其他命令插入。
  6. 高可用和分布式:通过Redis的Sentinel系统和Cluster集群模式,可以构建高可用、分布式的Redis系统。
     

redis安装与启动方法

在linux或者mac下安装redis是非常容易的,我们可以先下载redis的源码,然后make安装即可,不过Redis是用C语言实现的,要想编译,需要提前安装gcc等工具。编译redis的基本过程是:

wget http://download.redis.io/releases/redis-5.0.3.tar.gz
tar xzf redis-5.0.3.tar.gz
cd redis-5.0.3
make

如果没有安装wget,使用curl等工具也是可以的。

之后启动redis服务端的命令如下:

src/redis-server

这样就成功启动了Redis服务器。要连接到这个服务器,可以打开另一个终端窗口,然后输入:

 src/redis-cli

不过这样的话,使用的时候,需要先定位到安装位置,比较麻烦,为了方便使用,我们可以将上述地址配置到环境变量里,这样使用时只需要输入下面的命令即可:

redis-cli

如图所示:

这样我们就可以开心地使用Redis了。

2.Redis基础数据类型

在运行起来redis之后,接下来,我们的重点就是熟悉redis提供给我们的五种典型的数据结构,不同结构是针对不同的应用场景,我们这里分别介绍这几种结构的特征、用法、应用案例等等。对于用法,所有的结构,我们需要掌握的无非是“增删改查“以及判断长度、有无等等。

在工作中,我们一般通过java代码来调用redis提供的api接口来间接管理redis,但是这样比较繁琐,我们在后面会详细介绍如何使用这些api工具,这里我们先使用redis-cli来快速掌握redis相关的命令。

2.1 感受一下redis命令

在介绍redis的数据类型之前,我们先来感受一下如何通过命令来操作redis。redis命令的基本格式是:

KEYS pattern

pattern是通配符,这个与sql语言里的字符串匹配等是一样的, 常见的有:

符号含义
?匹配一个字符
*匹配人任意个(包括0个)字符
[]匹配括号间的任一字符,可以使用-表示范围,例如a[b-d]可以匹配"ab、ac、ad"这几种情况
\x匹配字符x,用于转义符号。如要匹配"?"就要使用“\?”

在打开redis-cli的窗口里,我们首先建立第一个键:

set haitao 1然后再输入:
KEYS *

此时的截图,由于我之前还有其他操作,内容会多一些:

如果我们想判断某个键是否存在,直接输入如下命令“exists haitao”即可:

如果返回结果是1表示存在,如果是0表示不存在。

如果想删除某个键, 使用“del key”命令即可,而且还可以删除多个键,后面返回的参数就是实际删除的键的个数,示例如下:

另外,我们还可以获取一下键的类型,常见的类型就是前面提到的五大类型:string(字符串类型)、hash(散列类型)、list(列表类型)、set(集合类型)、zset(有序集合),例如

2.2 字符串类型

String类型是Redis最基础的数据类型,是二进制安全的。它可以包含任何数据,例如jpeg图片或者序列化的对象,或者就简单的文本数据。String类型的value最大能存储512MB的数据。

字符串类型是其他4种数据类型的基础,其他数据类型和字符串类型的差别其实主要是组织字符串的方式不同,例如列表类型就是以列表的形式组织字符串等等。我们在后面会详细分析这几种数据类型的底层结构,会对此有更加深入的理解。

接下来,我们看一下字符串操作的常见命令。

1.SET:

SET命令用于设置指定键的值,如果键已经存在则覆盖旧的值。

SET name "haitao"

执行结果:OK。命令执行成功,则返回OK

2.GET:

GET命令用于获取指定键的值。

GET name

执行结果:"haitao"。如果键存在则返回其对应的值,否则返回nil

3.INCR:

INCR命令用于将指定键的值增加1。如果键不存在,那么在操作前会先将其设置为0,然后再执行加1操作。

SET counter 100
INCR counter

执行结果:101INCR命令返回执行操作后的值。

4. DECR:

DECR命令用于将指定键的值减少1。同样,如果键不存在,那么在操作前会先将其设置为0,然后再执行减1操作。

DECR counter

执行结果:100DECR命令返回执行操作后的值。

应用举例

在这里我们就想到redis的一种典型应用了:统计计数。例如我们想统计课程直播间实时在线人数,就可以使将数据先写到redis里,也就是:

set student_counter 100

redis里的数据不像数据表一样,相互之间独立,而是都在一个空间里,而在实际中,我们同时上课的直播间不止一个,我们可能还要统计某个直播间最高观看人数、当前在线人数、平均人数等等,所以我们会通过增加前缀的方式来区分,例如我们的规则是:统计的时候每个键按照如下规则设置:

科目Id_统计类型 人数

例如,采取如下的方式,就可以将各个业务的数据都存放起来了:

set math:101:max_student  100
set math:101:cur_student  50
set math:101:avg_student  60set english:102:max_student  80
set english:102:cur_student  50
set english:102:cur_student  40

正因为字符串的这个功能,我们还可以用来统计文章的访问量、课程的热度等等。另外我们可以还在微服务场景下,还可以redis来生成全局唯一的自增Id等重要功能。

2.3 List类型

List类型是Redis的一种复杂的数据类型,也是Redis的重要特性之一。Redis的List类型实际上是一个双向链表,所以添加一个新的元素到List的头部或者尾部的复杂度是O(1)。同时由于采用链表实现,可以利用索引下标来获取元素,或者进行范围查询,这个操作的复杂度是O(N)。

如果list中存储的元素非常多,我们可以想象到要获取首尾的元素仍然非常快,而要获取中间位置的元素则非常慢。这就决定了list适合处理”新鲜事、最新消息、最新评论、最新留言“这类的问题,
例如,在一个即时通讯应用中,我们一般都更关心最新的消息,因此可以使用List类型存储聊天记录,每次发送的消息都会添加到List中。
命令示例:

LPUSH chat:room1 "Hello"
LPUSH chat:room1 "How are you?"
LRANGE chat:room1 0 -1

这里的L就是left左侧的意思,一般理解为队列的头部,对应的还有个R就是right,右侧的意思,第对应的就是队列的尾部。push就是放入的意思,pop就是弹出元素的意思。通过不同的组合,我们可以构造出非常丰富的功能:

1. LPUSH:

LPUSH命令用于将一个或多个值插入到列表头部:

LPUSH mylist "World"
LPUSH mylist "Hello"

执行结果:2

LPUSH命令返回此时列表的长度。

执行命令:

LRANGE mylist 0 -1

查看列表,结果为:

127.0.0.1:6379> LRANGE mylist 0 -1
 1) "Hello"
 2) "World"

2. RPUSH:

RPUSH命令用于将一个或多个值插入到列表的尾部:

RPUSH mylist "Redis"

执行结果:3RPUSH命令返回此时列表的长度。

执行LRANGE mylist 0 -1命令查看列表,结果为:

1) "Hello"

2) "World"

3) "Redis"

3. LPOP:

LPOP命令用于移除并返回列表的第一个元素:

LPOP mylist

执行结果:"Hello"LPOP命令返回被移除的元素的值。

执行LRANGE mylist 0 -1命令查看列表,结果为:

1) "World"

2) "Redis"

4. RPOP:

RPOP命令用于移除并返回列表的最后一个元素:

RPOP mylist

执行结果:"Redis"RPOP命令返回被移除的元素的值。

执行LRANGE mylist 0 -1命令查看列表,结果为:1) "World"

以上即为Redis List类型的常用命令示例,以及它们的执行结果。 

2.4 Set类型

Set类型也是Redis的一种复杂数据类型,它是一种无序的、元素不重复的集合。由于集合中的元素是唯一的,所以可以使用Set类型来存储一些集合元素,并进行交集、并集、差集等操作。

对于set的应用,最典型的是记录社交网络中的用户之间的关系,可以使用Set类型存储用户的好友列表,以便进行好友关系的管理。
命令示例:

SADD user:1:friends user:2
SADD user:1:friends user:3
SMEMBERS user:1:friends

可以看到此时可以找到user1的两个朋友user2和user3。同样在线培训系统中,老师与课程、课程与学生、老师与学生一般都是多对多的复杂关系,我们可以采用上面方式来轻松记录。

常用命令示例

1. SADD:

SADD命令用于向集合添加一个或多个成员。

SADD myset "Hello"
SADD myset "World"

执行结果:1SADD命令返回添加成功的元素数量。

执行SMEMBERS myset命令查看集合,结果为:

1) "Hello"

2) "World"

2. SREM:

SREM命令用于移除集合中一个或多个成员。

SREM myset "World"

执行结果:1SREM命令返回移除成功的元素数量。

执行SMEMBERS myset命令查看集合,结果为:

1) "Hello"

3. SISMEMBER:

SISMEMBER命令用于判断成员元素是否是集合的成员。

SISMEMBER myset "World"
SISMEMBER myset "Hello"

执行结果:0 和 1SISMEMBER命令返回元素是否在集合中,0表示不在,1表示在。

2.5 Hash类型

Hash类型是Redis的一种复杂数据类型,它相当于一种简单的二级字典:第一级的key对应一个新的字典,这个新的字典就是Hash的内部结构,拥有自己的key-value对。Hash类型适合用于存储对象。

Hash结构虽然略复杂,但确是我们使用redis进行缓存的核心结构,因为缓存的内容,我们经常需要根据键来快速获取某个特定的对象。例如某个网站需要存储用户的个人信息,可以使用Hash类型存储用户的姓名、年龄、性别等字段。
命令示例:

HSET user:1 name "haitao"
HSET user:1 age 25
HSET user:1 gender "Male"
HGETALL user:1

常用命令示例

1. HSET:

HSET命令用于为哈希表中的字段赋值。

HSET myhash field1 "Hello"
HSET myhash field2 "World"

执行结果:1HSET命令返回操作成功的数量。

执行HGETALL myhash命令查看哈希表,结果为:

1) "field1"

2) "Hello"

3) "field2"

4) "World"

2. HGET:

HGET命令用于获取存储在哈希表中指定字段的值。

HGET myhash field1

执行结果:"Hello"HGET命令返回字段的值。

3. HGETALL:

HGETALL命令用于获取在哈希表中指定key的所有字段和值。

HGETALL myhash

2.6 Sorted Set类型

Sorted Set是Redis的一种复杂数据类型,也是集合的一种,能够存储一组数据,并且每个数据都有一个得分。数据是唯一的,但是得分(score)可以重复。集合是通过哈希表实现的,所以添加,删除,查找的复杂度都是O(1)。而得分的值是通过跳跃表来进行排序的。

有序set最典型的场景是热榜,例如微博可以使用Sorted Set类型存储热门新闻的排名,以便展示热门文章;在培训系统中可以根据点击量来判断哪些课时或者内容更受欢迎等等。命令示例:

ZADD news:popular 1000 "Article1"
ZADD news:popular 800 "Article2"
ZADD news:popular 600 "Article3"
ZREVRANGE news:popular 0 -1 WITHSCORES

常用命令示例

1. ZADD:

ZADD命令用于向有序集合添加一个或多个成员,或者更新已存在成员的分数。

ZADD myzset 1 "one"
ZADD myzset 2 "two" 3 "three"

执行结果:1。ZADD命令返回被成功添加的新成员的数量。

执行

ZRANGE myzset 0 -1 WITHSCORES

命令查看有序集合,结果为:

127.0.0.1:6379> ZRANGE myzset 0 -1 WITHSCORES

1) "one"

2) "1"

3) "two"

4) "2"

5) "three"

6) "3"

2. ZSCORE:

ZSCORE命令返回有序集中,成员的分数值。

ZSCORE myzset "one"

执行结果:"1"ZSCORE命令返回成员的分数值。

3. ZRANGE:

ZRANGE命令用于返回有序集中,指定区间内的成员。

ZRANGE myzset 0 -1

本文,我们介绍了redis最重要的五种数据类型的特征、使用方法和典型应用场景,为我们后面的学习打下坚实的基础! 

相关文章:

【征服redis1】基础数据类型详解和应用案例

博客计划 ,我们从redis开始,主要是因为这一块内容的重要性不亚于数据库,但是很多人往往对redis的问题感到陌生,所以我们先来研究一下。 本篇,我们先看一下redis的基础数据类型详解和应用案例。 1.redis概述 以mysql为…...

【WPF.NET开发】WPF中的XAML资源

本文内容 使用 XAML 中的资源静态和动态资源静态资源动态资源样式、DataTemplate 和隐式键 资源是可以在应用中的不同位置重复使用的对象。 资源的示例包括画笔和样式。 本概述介绍如何使用 Extensible Application Markup Language (XAML) 中的资源。 你还可以使用代码创建和…...

ChatGPT 淘金潮(全)

原文:The ChatGPT GoldRush 译者:飞龙 协议:CC BY-NC-SA 4.0 一、ChatGPT 简介 什么是 ChatGPT? ChatGPT 是由 OpenAI 基于 GPT-4 架构创建的大型语言模型。它旨在理解和回应自然语言文本输入,使得可以与机器进行对话…...

【零基础入门Python数据分析】Anaconda3 JupyterNotebookseaborn版

目录 一、安装环境 python介绍 anaconda介绍 jupyter notebook介绍 anaconda3 环境安装 解决JuPyter500:Internal Server Error问题-CSDN博客 Jupyter notebook快捷键操作大全 二、Python基础入门 数据类型与变量 数据类型 变量及赋值 布尔类型与逻辑运算…...

C++面试:单例模式、工厂模式等简单的设计模式 创建型、结构型、行为型设计模式的应用技巧

理解和能够实现基本的设计模式是非常重要的。这里,我们将探讨两种常见的设计模式:单例模式和工厂模式,并提供一些面试准备的建议。 目录 单例模式 (Singleton Pattern) 工厂模式 (Factory Pattern) 面试准备 1. 理解设计模式的基本概念…...

Oracle JDK 8 中的 computeIfAbsent 方法及实践

Java 8 引入了一系列新特性,其中之一是对 Map 接口的增强,其中包括了 computeIfAbsent 方法。这个方法为处理映射提供了一种便捷而强大的方式,允许在键不存在或对应的值为 null 时,动态计算新的值并将其放入映射。在本篇博客中&am…...

华为设备vlan下配置MSTP,STP选举

核心代码,不同实例,承载不同流量,为每个实例设置一个根网桥达到分流的效果 stp region-config //进入stp区域的设置 region-name R1 //区域命名为R1 instance 1 vlan 10 …...

案例学Python:filter()函数的用法,高级!

大家好,这里是程序员晚枫,又来分享有用的Python知识了。 Python之所以好用,是因为有大量用于科学计算的内置函数和第三方库。用好这些第三方库,可以显著提高我们编程的速度和质量。 今天我们一起来看一下Python中一个重要的内置…...

jmeter--7.BeanShell

目录 1. beanshell常用语法 1.1 log:日志写入 1.2 vars:设置和引用局部变量(同线程组) 1.3 props:设置和引用全局变量(跨线程组) 1.4 prev:获取前一个请求返回的信息 2. beans…...

第 3 场 蓝桥杯小白入门赛 解题报告 | 珂学家 | 单调队列优化的DP + 三指针滑窗

前言 整体评价 T5, T6有点意思&#xff0c;这场小白入门场&#xff0c;好像没真正意义上的签到&#xff0c;整体感觉是这样。 A. 召唤神坤 思路: 前后缀拆解 #include <iostream> #include <algorithm> #include <vector> using namespace std;int main()…...

debian apt 装 mysql8

MySQL &#xff1a;&#xff1a; MySQL 8.0 参考手册 &#xff1a;&#xff1a; 2.5.5 使用来自 Oracle 的 Debian 软件包在 Linux 上安装 MySQL apt install -f lsb-release gnupg wget https://repo.mysql.com//mysql-apt-config_0.8.29-1_all.deb dpkg -i mysql-apt-config…...

LeetCode 每日一题 Day 37-43

终于考完试了&#xff0c;寒假期间将会每天持续更新&#xff01; 447. 回旋镖的数量(Day 37) 给定平面上 n 对 互不相同 的点 points &#xff0c;其中 points[i] [xi, yi] 。回旋镖 是由点 (i, j, k) 表示的元组 &#xff0c;其中 i 和 j 之间的欧式距离和 i 和 k 之间的欧…...

产品百度百科怎么创建?产品如何上百度百科?

百度百科作为一个权威的信息平台&#xff0c;承载着巨大的流量和曝光度。对于一个产品来说&#xff0c;能够在百度百科上拥有一席之地&#xff0c;无疑是一种极高的荣誉&#xff0c;同时也是提升品牌知名度、增加信任度的重要手段。产品百度百科不仅能够详细、全面地介绍产品信…...

Vue keep-alive的使用和原理解析

✨ 专栏介绍 在当今Web开发领域中&#xff0c;构建交互性强、可复用且易于维护的用户界面是至关重要的。而Vue.js作为一款现代化且流行的JavaScript框架&#xff0c;正是为了满足这些需求而诞生。它采用了MVVM架构模式&#xff0c;并通过数据驱动和组件化的方式&#xff0c;使…...

Fine-tuning:个性化AI的妙术

在本篇文章中,我们将深入探讨Fine-tuning的概念、原理以及如何在实际项目中运用它,以此为初学者提供一份入门级的指南。 一、什么是大模型 ChatGPT大模型今年可谓是大火,在正式介绍大模型微调技术之前,为了方便大家理解,我们先对大模型做一个直观的抽象。 本质上,现在…...

说清楚Kubernetes、Docker、Dockershim、Containerd、runC、CRI、OCI的关系

Kubernetes v1.20版本 的 release note 里说 deprecated docker。并且在后续版本 v1.24 正式删除了 dockershim 组件&#xff0c;这对我们有什么影响呢&#xff1f;Kubernetes 1.20: The Raddest Release | Kubernetes 为了搞明白这件事情&#xff0c;以及理解一系列容器名词 …...

x-cmd pkg | trash-cli - 类 Unix 系统的命令行垃圾桶

目录 简介首次用户技术特点竞品和相关作品进一步阅读 简介 trash-cli 是类 Unix 系统的命令行垃圾桶&#xff0c;用于移动文件到回收站&#xff0c;同时会记录文件的原地址和删除日期。 该工具使用与 GNOME、KDE ​​和 XFCE 等桌面环境相同的垃圾桶&#xff0c;所以即使是非 …...

基于Java+SpringBoot+vue实现图书借阅和销售商城一体化系统

基于JavaSpringBootvue实现图书借阅和销售商城一体化系统 &#x1f345; 作者主页 央顺技术团队 &#x1f345; 欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; &#x1f345; 文末获取源码联系方式 &#x1f4dd; &#x1f345; 查看下方微信号获取联系方式 承接各种定制系…...

集成xxljob项目如何迁移到K8S

前言 大家好&#xff0c;今天我们将基于XXL-Job&#xff0c;探讨任务调度迁移到云端的相关话题。 XXL-Job是一款功能强大、易用可靠的国产分布式任务调度平台&#xff0c;是目前国内使用比较广泛的分布式任务调度平台之一。它的主要特点包括&#xff1a; 支持分布式、多线程…...

类型“{}”上不存在属性“xxx”。ts(2339)-解决方案集锦

类型“{}”上不存在属性“xxx”。ts(2339)-解决方案集锦 文章目录 类型“{}”上不存在属性“xxx”。ts(2339)-解决方案集锦一、方案一&#xff08;优先尝试&#xff09;二、方案二&#xff08;优先尝试&#xff09;三、方案三这该是多么痛苦的一篇笔记啊&#xff01;&#xff0…...

聊聊 Pulsar:Producer 源码解析

一、前言 Apache Pulsar 是一个企业级的开源分布式消息传递平台&#xff0c;以其高性能、可扩展性和存储计算分离架构在消息队列和流处理领域独树一帜。在 Pulsar 的核心架构中&#xff0c;Producer&#xff08;生产者&#xff09; 是连接客户端应用与消息队列的第一步。生产者…...

Leetcode 3577. Count the Number of Computer Unlocking Permutations

Leetcode 3577. Count the Number of Computer Unlocking Permutations 1. 解题思路2. 代码实现 题目链接&#xff1a;3577. Count the Number of Computer Unlocking Permutations 1. 解题思路 这一题其实就是一个脑筋急转弯&#xff0c;要想要能够将所有的电脑解锁&#x…...

全球首个30米分辨率湿地数据集(2000—2022)

数据简介 今天我们分享的数据是全球30米分辨率湿地数据集&#xff0c;包含8种湿地亚类&#xff0c;该数据以0.5X0.5的瓦片存储&#xff0c;我们整理了所有属于中国的瓦片名称与其对应省份&#xff0c;方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...

[10-3]软件I2C读写MPU6050 江协科技学习笔记(16个知识点)

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16...

零基础在实践中学习网络安全-皮卡丘靶场(第九期-Unsafe Fileupload模块)(yakit方式)

本期内容并不是很难&#xff0c;相信大家会学的很愉快&#xff0c;当然对于有后端基础的朋友来说&#xff0c;本期内容更加容易了解&#xff0c;当然没有基础的也别担心&#xff0c;本期内容会详细解释有关内容 本期用到的软件&#xff1a;yakit&#xff08;因为经过之前好多期…...

Redis的发布订阅模式与专业的 MQ(如 Kafka, RabbitMQ)相比,优缺点是什么?适用于哪些场景?

Redis 的发布订阅&#xff08;Pub/Sub&#xff09;模式与专业的 MQ&#xff08;Message Queue&#xff09;如 Kafka、RabbitMQ 进行比较&#xff0c;核心的权衡点在于&#xff1a;简单与速度 vs. 可靠与功能。 下面我们详细展开对比。 Redis Pub/Sub 的核心特点 它是一个发后…...

MySQL 知识小结(一)

一、my.cnf配置详解 我们知道安装MySQL有两种方式来安装咱们的MySQL数据库&#xff0c;分别是二进制安装编译数据库或者使用三方yum来进行安装,第三方yum的安装相对于二进制压缩包的安装更快捷&#xff0c;但是文件存放起来数据比较冗余&#xff0c;用二进制能够更好管理咱们M…...

宇树科技,改名了!

提到国内具身智能和机器人领域的代表企业&#xff0c;那宇树科技&#xff08;Unitree&#xff09;必须名列其榜。 最近&#xff0c;宇树科技的一项新变动消息在业界引发了不少关注和讨论&#xff0c;即&#xff1a; 宇树向其合作伙伴发布了一封公司名称变更函称&#xff0c;因…...

STM32---外部32.768K晶振(LSE)无法起振问题

晶振是否起振主要就检查两个1、晶振与MCU是否兼容&#xff1b;2、晶振的负载电容是否匹配 目录 一、判断晶振与MCU是否兼容 二、判断负载电容是否匹配 1. 晶振负载电容&#xff08;CL&#xff09;与匹配电容&#xff08;CL1、CL2&#xff09;的关系 2. 如何选择 CL1 和 CL…...

热烈祝贺埃文科技正式加入可信数据空间发展联盟

2025年4月29日&#xff0c;在福州举办的第八届数字中国建设峰会“可信数据空间分论坛”上&#xff0c;可信数据空间发展联盟正式宣告成立。国家数据局党组书记、局长刘烈宏出席并致辞&#xff0c;强调该联盟是推进全国一体化数据市场建设的关键抓手。 郑州埃文科技有限公司&am…...