SpringBoot 项目整合 Redis 教程详解
✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。
🍎个人主页:Java Fans的博客
🍊个人信条:不迁怒,不贰过。小知识,大智慧。
💞当前专栏:SpringBoot 框架从入门到精通
✨特色专栏:国学周更-心性养成之路
🥭本文内容:SpringBoot 项目整合 Redis 教程详解
文章目录
- Redis 的优势
- Redis 安装
- Redis 数据类型
- 1.String(字符串)
- 2.Hash(哈希)
- 3.List(列表)
- 4.Set(集合)
- 5.ZSet(sorted set:有序集合)
- SpringBoot 操作 Redis
- SpringBoot 配置 Redis
- RedisTemplate 及其相关方法
- 1.RedisTemplate 介绍
- 2.Redis 5种数据结构操作
- Redis 业务实践
Redis 是完全开源的,遵守 BSD 协议,是一个高性能的 key-value 数据库.
Redis 与其他 key - value 缓存产品有以下三个特点:
- Redis支持数据的持久化,可以将内存中的数据保存在磁盘中,重启的时候可以再次加载进行使用。
- Redis不仅仅支持简单的key-value类型的数据,同时还提供list,set,zset,hash等数据结构的存储。
- Redis支持数据的备份,即master-slave模式的数据备份。
Redis 的优势
- 性能极高 – Redis能读的速度是110000次/s,写的速度是81000次/s
- 丰富的数据类型 – Redis支持二进制案例的 String, List, Hash, Set 及 zset数据类型操作。
- 原子 – Redis的所有操作都是原子性的,意思就是要么成功执行要么失败完全不执行。单个操作是原子性的。多个操作也支持事务,即原子性,通过MULTI和EXEC指令包起来。
- 丰富的特性 – Redis还支持 publish/subscribe, 通知, key 过期等等特性
- Redis 是单线程的,6.0版本开始支持开启多线程。
Redis 安装
下载地址: https://github.com/tporadowski/redis/releases。
解压下载后的压缩文件,解压后文件列表如下:
使用cmd窗口打开Redis
redis-server.exe redis.windows.conf #加载配置文件启动
注:启动之后,不要关闭窗口,关闭窗口服务停止!
安装Redis数据库客户端
库相关指令:
flushdb 清空当前库flushall 清空所有库select 1 切换库
key的相关指令
指令 | 作用 | 语法 |
---|---|---|
del | 删除一个或多个key | del keyname |
exists | 判断一个或多个key是否存在,多个key时有一个存在则就会返回1 | exists keyname |
expire | 设置key的生存时间 单位 :秒 | expire keyname seconds |
keys | 查询所有匹配模式的key ?匹配一个字符 *匹配0-n个字符 [] 满足其中的一个 | key * key h?llo |
move | 将key移动到指定的库中 | move keyname db |
pexpire | 设置key的生存时间 单位 :毫秒 设置成功返回1 否则返回0 | pexpire keyname milliseconds |
ttl | 以秒为单位返回key的剩余生存时间,返回-1表示永久存储,-2表示key不存在 | ttl keyname |
randomkey | 从当前数据库中随机的返回一个key | randomkey |
rename | 重命名key,成功返回ok,否则返回错误信息。 | rename key newkey |
type | 返回key所存储的值的类型 | type keyname |
Redis 数据类型
1.String(字符串)
- string 是 redis 最基本的类型,你可以理解成与 Memcached 一模一样的类型,一个 key 对应一个 value。
- string 类型是二进制安全的。意思是 redis 的 string 可以包含任何数据。比如jpg图片或者序列化的对象。
- string 类型是 Redis 最基本的数据类型,string 类型的值最大能存储 512MB。
操作指令:
命令 | 描述 |
---|---|
SET | 设置指定 key 的值 |
GET | 获取指定 key 的值。 |
GETRANGE | 返回 key 中字符串值的子字符 |
GETSET | 将给定 key 的值设为 value ,并返回 key 的旧值(old value)。 |
SETEX | 将值 value 关联到 key ,并将 key 的过期时间设为 seconds (以秒为单位)。 |
SETNX | 只有在 key 不存在时设置 key 的值 |
STRLEN | 返回 key 所储存的字符串值的长度。 |
MSET | 同时设置一个或多个 key-value 对。 |
MSETNX | 同时设置一个或多个 key-value 对,当且仅当所有给定 key 都不存在 |
INCR | 将 key 中储存的数字值增一 |
INCRBY | 将 key 所储存的值加上给定的增量值(increment) |
INCRBYFLOAT | 将 key 所储存的值加上给定的浮点增量值(increment) |
DECR | 将 key 中储存的数字值减一。 |
DECRBY | key 所储存的值减去给定的减量值(decrement) |
APPEND | 如果 key 已经存在并且是一个字符串, APPEND 命令将指定的 value 追加到该 key 原来值(value)的末尾 |
2.Hash(哈希)
- Redis hash 是一个键值(key=>value)对集合。
- Redis hash 是一个 string 类型的 field 和 value 的映射表,hash 特别适合用于存储对象。
操作指令:
命令 | 描述 |
---|---|
hset | 设置一个key/value对 |
hget | 获取key对应的value |
hgetall | 获取所有的key/value对 |
hdel | 删除某个key/value对 |
hexists | 判断一个key是否存在 |
hkeys | 获取所有的key |
hvals | 获取所有的value |
hmset | 设置多个key/value |
hmget | 获取多个key的value |
hsetnx | 设置一个不存在的key的值 |
hincrby | 为value的值进行加法运算 |
hincrbyfloat | 为value的值进行加浮点类型值运算 |
3.List(列表)
- Redis 列表是简单的字符串列表,按照插入顺序排序。你可以添加一个元素到列表的头部(左边)或者尾部(右边)。
操作指令
命令 | 描述 |
---|---|
LINDEX | 通过索引获取列表中的元素 lindex lists 0 |
LINSERT key BEFORE|AFTER | 在列表的元素前或者后插入元素 |
LLEN | 获取列表长度 |
LPOP | 移出并获取列表的第一个元素 |
LPUSH | 将一个或多个值插入到列表头部 |
LPUSHX | 将一个值插入到已存在的列表头部 |
LRANGE | 获取列表指定范围内的元素 (0 -1) |
LREM | 移除列表重复元素 |
LSET | 通过索引设置列表元素的值 ,但是索引必须存在,实质是根据索引修改值 |
LTRIM | 对一个列表进行修剪(trim),就是说,让列表只保留指定区间内的元素,不在指定区间之内的元素都将被删除 |
RPOP | 移除列表的最后一个元素,返回值为移除的元素 |
RPOPLPUSH | 移除列表的最后一个元素,并将该元素添加到另一个列表并返回 |
RPUSH | 在列表中添加一个或多个值 |
RPUSHX | 为已存在的列表添加值 |
4.Set(集合)
- Redis 的 Set 是 string 类型的无序集合。
- 集合是通过哈希表实现的,所以添加,删除,查找的复杂度都是 O(1)。
操作指令:
命令 | 描述 |
---|---|
sadd | 为集合添加元素 |
smembers | 显示集合中所有元素 (无序) |
scard | 返回集合中元素的个数 |
spop | 随机返回一个元素,并将这个元素删除 |
smove | 从一个集合向令一个集合中转移元素 |
srem | 从集合中删除一个元素 |
sismember | 判断集合中是否包含这个元素 |
srandmember | 随机返回一个元素 |
sinter | 求交集 |
sunion | 求和集 |
5.ZSet(sorted set:有序集合)
- Redis ZSet 和 Set 一样也是 String 类型元素的集合,且不允许重复的成员。
- 不同的是每个元素都会关联一个 double 类型的分数。Redis 正是通过分数来为集合中的成员进行从小到大的排序。
- ZSet 的成员是唯一的,但分数(score)却可以重复。
操作指令:
命令 | 描述 |
---|---|
zadd | 添加一个有序集合元素 |
zcard | 返回集合中元素的个数 |
zrange升序 zrevrange降序 | 返回一个范围内的元素 |
zrangebyscore | 按照分数查找一个范围内的元素 |
zrank | 返回排名 |
zrevrank | 倒叙排名 |
zscore | 显示某个元素的分数 |
zrem | 移除某个元素 |
zincrby | 给某个特定元素加分 |
SpringBoot 操作 Redis
spring boot data redis中提供了RedisTemplate和StringRedisTemplate,其中StringRedisTemplate是Redistemplate的子类,两个方法基本一致,不同之处主要体现在操作的数据类型不同,RedisTemplate中的两个泛型都是Object,意味着存储的key和value都可以是一个对象,而StringRedisTemplate的两个泛型都是String,意味着StringRedisTemplate的key和value都只能是字符串。
引入依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
SpringBoot 配置 Redis
spring:redis:# Redis数据库索引(默认为0)database: 0# Redis服务器地址host: 127.0.0.1# Redis服务器连接端口port: 6379# Redis服务器连接密码(默认为空)password:# 连接池最大连接数(使用负值表示没有限制)jedis.pool.max-active: 20# 连接池最大阻塞等待时间(使用负值表示没有限制)jedis.pool.max-wait: -1# 连接池中的最大空闲连接jedis.pool.max-idle: 10# 连接池中的最小空闲连接jedis.pool.min-idle: 0# 连接超时时间(毫秒)timeout: 1000
RedisTemplate 及其相关方法
1.RedisTemplate 介绍
Spring封装了RedisTemplate对象来进行对Redis的各种操作,它支持所有的Redis原生的api。RedisTemplate位于spring-data-redis包下。RedisTemplate提供了redis各种操作、异常处理及序列化,支持发布订阅。
2.Redis 5种数据结构操作
- redisTemplate.opsForValue(); //操作字符串
- redisTemplate.opsForHash(); //操作hash
- redisTemplate.opsForList(); //操作list
- redisTemplate.opsForSet(); //操作set
- redisTemplate.opsForZSet(); //操作有序set
或者:
- redistempalate.boundValueOps
- redistempalate.boundSetOps
- redistempalate.boundListOps
- redistempalate.boundHashOps
- redistempalate.boundZSetOps
opsForXXX和boundXXXOps的区别:XXX为value的类型,前者获取一个operator,但是没有指定操作的对象(key),可以在一个连接(事务)内操作多个key以及对应的value;后者获取了一个指定操作对象(key)的operator,在一个连接(事务)内只能操作这个key对应的value。
SpringBootTest 实现Redis数据库增删改查
/*** 使用RedisTemplate 操作Redis数据的不同数据类型*/
@SpringBootTest
public class Springbootday03ApplicationTests {@Autowiredprivate RedisTemplate<String, String> redisTemplate;/*** String 类型数据操作*/@Testpublic void operateString() {//添加值redisTemplate.opsForValue().set("str", "strValue1");//添加值 判定是否存在 存在则不添加Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent("str", "strAbsent");System.out.println("str设置成功:" + aBoolean);//获取值String str = redisTemplate.opsForValue().get("str");System.out.println("str = " + str);//更新值redisTemplate.opsForValue().set("str", "strValue2");str = redisTemplate.opsForValue().get("str");System.out.println("newStr = " + str);//删除值Boolean b = redisTemplate.delete("str");System.out.println("str删除成功:" + b);}/*** 操作string类型数据 设置过期时间*/@Testpublic void operateString2() {redisTemplate.opsForValue().set("str", "strTimeout", 10, TimeUnit.SECONDS);//判定值是否存在 不存在则设置值 同时设置过期时间Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent("str2", "strTimeoutAbsent", 20, TimeUnit.SECONDS);System.out.println("setIfAbsent:" + aBoolean);}/*** 操作hash类型数据*/@Testpublic void operateHash() {//添加hash类型数据 key - valueredisTemplate.opsForHash().put("hash", "username", "admin");//修改hash类型数据redisTemplate.opsForHash().put("hash", "username", "tom");redisTemplate.opsForHash().put("hash", "password", "123456");//添加hash类型数据 key - mapHashMap<String, String> map = new HashMap<>();map.put("driverName", "com.mysql.jdbc.Driver");map.put("url", "jdbc:mysql://localhost:3306/mybatis?serverTimezone=UTC");redisTemplate.opsForHash().putAll("hash", map);//获取hash类型数据 entriesMap<Object, Object> hash = redisTemplate.opsForHash().entries("hash");hash.forEach((key, value) -> {System.out.println(key + "::" + value);});//获取所有的keySet<Object> keys = redisTemplate.opsForHash().keys("hash");for (Object key : keys) {System.out.println("key:" + key);}//获取所有valueList<Object> values = redisTemplate.opsForHash().values("hash");values.forEach(value -> System.out.println("value:" + value));//删除hash类型数据 删除一个 返回删除的个数Long delete = redisTemplate.opsForHash().delete("hash", "username");System.out.println("delete = " + delete);//删除hash类型数据 删除多个 返回删除的个数delete = redisTemplate.opsForHash().delete("hash", "username", "password", "driverName");System.out.println("delete = " + delete);//删除hash类型数据 删除所有Boolean delHash = redisTemplate.delete("hash");System.out.println("delHah:" + delHash);}/*** 操作List类型 有序 可重复*/@Testpublic void operateList() {//左压栈// redisTemplate.opsForList().leftPush("list", "listValue1");// redisTemplate.opsForList().leftPush("list", "listValue1");// redisTemplate.opsForList().leftPush("list", "listValue2");// redisTemplate.opsForList().leftPush("list", "listValue3");//右压栈redisTemplate.opsForList().rightPush("list", "listValue0");redisTemplate.opsForList().rightPush("list", "listValue2");redisTemplate.opsForList().rightPush("list", "listValue0");//左出栈String list1 = redisTemplate.opsForList().leftPop("list");System.out.println("leftPop list1 = " + list1);//右出栈String list2 = redisTemplate.opsForList().rightPop("list");System.out.println("rightPop list2 = " + list2);//获取所有数据List<String> lists = redisTemplate.opsForList().range("list", 0, redisTemplate.opsForList().size("list") - 1);lists.forEach(list -> System.out.println(list));//设置指定位置的数据redisTemplate.opsForList().set("list", 0, "listValue0");/*** 从存储在键中的列表中删除等于值的元素的第一个计数事件。* count> 0:删除等于从左到右移动的值的第一个元素;* count< 0:删除等于从右到左移动的值的第一个元素;* count = 0:删除等于value的所有元素。*/Long remove = redisTemplate.opsForList().remove("list", -1, "listValue0");System.out.println("remove:" + remove);//删除指定key的list数据Boolean list = redisTemplate.delete("list");System.out.println("list集合删除成功:" + list);}/*** 操作Set类型 无序 不可重复*/@Testpublic void operateSet() {//设置set值redisTemplate.opsForSet().add("set", "setValue0");redisTemplate.opsForSet().add("set", "setValue0");redisTemplate.opsForSet().add("set", "setValue1");//判定是否包含Boolean member = redisTemplate.opsForSet().isMember("set", "setValue0");System.out.println("isMember:" + member);//删除set中的值Long remove = redisTemplate.opsForSet().remove("set", "setValue0");System.out.println("remove = " + remove);//获取set类型值Set<String> set = redisTemplate.opsForSet().members("set");set.forEach(str -> {System.out.println("str = " + str);});}/*** 操作 ZSet 有序 不可重复*/@Testpublic void operateZSet() {//存储值Boolean add = redisTemplate.opsForZSet().add("zset", "zsetValue0", 10);System.out.println("add = " + add);System.out.println("add = " + add);add = redisTemplate.opsForZSet().add("zset", "zsetValue2", 2);System.out.println("add = " + add);//获取值// Boolean zset = redisTemplate.delete("zset");// System.out.println("delete zset = " + zset);}
}
Redis工具类的封装
/*** Redis 工具类* @author mosin* date 2021/11/30* @version 1.0*/
@Component
public final class RedisUtil {private RedisUtil(){};@Autowiredprivate RedisTemplate<String,String> redisTemplate;//设置值public void setValue(String key,String value){redisTemplate.opsForValue().set(key, value);}// 设置值 同时设置有效时间public void setValue(String key, String value, Long timeOut, TimeUnit timeUnit){redisTemplate.opsForValue().setIfAbsent(key, value, timeOut, timeUnit);}//设置值 没有则设置 有则不设置public void setNx(String key,String value){redisTemplate.opsForValue().setIfAbsent(key, value);}//设置值 没有则设置 同时设置有效时间 有则不设置public void setNx(String key,String value,long timeOut,TimeUnit timeUnit){redisTemplate.opsForValue().setIfAbsent(key, value,timeOut,timeUnit);}//删除值public boolean del(String key){return redisTemplate.delete(key);}//获取值public String getValue(String key){return redisTemplate.opsForValue().get(key);}
}
Redis 业务实践
redis 存储 token,实现非法请求拦截
1.编写拦截器
@Component
public class AdminInterceptor implements HandlerInterceptor {@Autowiredprivate RedisUtil redisUtil;@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("拦截器以拦截请求");//从请求头中获取token 验证用户是否登录String token = request.getHeader("token");System.out.println(token);String tokenValue = redisUtil.getValue(token);System.out.println("tokenValue = " + tokenValue);if(tokenValue!=null){ //用户已登录 放行请求return true;}else{//重定向到登录页面response.sendRedirect(request.getContextPath()+"/login.jsp");return false;}}
}
2.配置拦截器
@Configuration
public class LoginConfig implements WebMvcConfigurer {@Autowiredprivate AdminInterceptor adminInterceptor;@Overridepublic void addInterceptors(InterceptorRegistry registry) {InterceptorRegistration registration = registry.addInterceptor(adminInterceptor);registration.addPathPatterns("/**");registration.excludePathPatterns("/user/login","/user/register","/login.jsp");}
}
3.编写统一返回数据格式类
@Data
@AllArgsConstructor
@NoArgsConstructor
@Builder
public class JsonResult<T> {private Integer code;private String msg;private Long count;private T data;
}
4.编写控制器
@Controller
@RequestMapping("/user")
public class UserController {@Autowiredprivate UserService userService;@Autowiredprivate RedisUtil redisUtil;@ResponseBody@RequestMapping("/login")public Object login(User user) throws JsonProcessingException {User usr = User.builder().id(1).name("admin").password("123456").build();//获取token 放入redisString token = UUID.randomUUID().toString().replace("-", "");//将user 转为json格式放入 redisObjectMapper objectMapper = new ObjectMapper();String s1 = objectMapper.writeValueAsString(usr);//将 token 和用户信息存入 redisredisUtil.setValue(token, s1, 2L, TimeUnit.MINUTES);//将token 存入map集合返回HashMap<String, String> map = new HashMap<>();map.put("token", token);return map;}@ResponseBody@RequestMapping("/register")public Object register(User user){HashMap<String, String> map = new HashMap<>();map.put("msg", "ok");return map;}@ResponseBody@RequestMapping("/add")public Object add(User user){HashMap<String, String> map = new HashMap<>();map.put("msg", "ok");return map;}
}
5.编写业务类和Mapper接口
6.使用postman接口测试工具测试接口
码文不易,本篇文章就介绍到这里,如果想要学习更多Java系列知识,点击关注博主,博主带你零基础学习Java知识。与此同时,对于日常生活有困扰的朋友,欢迎阅读我的第四栏目:《国学周更—心性养成之路》,学习技术的同时,我们也注重了心性的养成。
相关文章:

SpringBoot 项目整合 Redis 教程详解
✅作者简介:2022年博客新星 第八。热爱国学的Java后端开发者,修心和技术同步精进。 🍎个人主页:Java Fans的博客 🍊个人信条:不迁怒,不贰过。小知识,大智慧。 💞当前专栏…...

3ASC25H214 DATX130以力控制为基础的装配应用方面已经形成了一个解决方案
3ASC25H214 DATX130以力控制为基础的装配应用方面已经形成了一个解决方案 ABB的机器人解决方案最终选择了IRB6400机器人 ABB的解决方案 ABB一直都在不断地研究和开发机器人应用的新技术,有一部分研究活动是与大学进行合作的,其中一项是ABB的科学家和…...

Java的位运算
目录 1 Java中支持的位运算 2 位运算规则 3 逻辑运算 3.1 与运算(&) 3.2 或运算(|) 3.3 异或运算(^) 3.3 取反运算(~) 4 位移操作 4.1 左移(<<&#…...

FastDFS分布式文件存储
FastDFS文件上传 简介: 主要解决:大容量的文件存储和高并发访问的问题 论坛:https://bbs.chinaunix.net 下载网站:https://sourceforge.net/projects/fastdfs/files/ 安装参考:https://www.cnblogs.com/cxygg/p/1…...

Android的AAC架构
AAC Android Architecture Components的简称,是一套用来搭建具有生命周期感知架构的系列组件,在2017年 GoogleI/O大会上发布。 dependencies {def lifecycle_version "2.2.0"implementation "androidx.lifecycle:lifecycle-livedata-ktx…...

高功率激光切割中不良现象的排除技巧
高功率切割市场现状 随着激光行业的发展和下游产业需求的变化,高功率的激光切割设备已逐渐成为市场关注的热点。高功率激光切割凭着速度和厚度上无可比拟的优势,目前已获得了市场的广泛认可。 但由于高功率激光切割技术尚处于普及的初级阶段,…...

MySQL-----复合查询
文章目录 前言一、基本查询回顾二、 多表查询解决多表查询的思路 三、自连接四、子查询1. 单行子查询2. 多行子查询3. 多列子查询4. 在from子句中使用子查询5. 合并查询5.1 union5.2 unoin all 总结 前言 前面的学习中,对于mysql表的查询都是对一张表进行查询,在实际开发中这远…...

10.Yarn概述
如果说HDFS是存储,则Yarn就是cpu和内存,mapreduce就是程序。 1.基础架构 复习: 1.Container就是一个容器,其中封装了需要使用的内存与cpu 2.每当提交一个job,就会产生一个appMaster(总指挥),app Master负责其他container里面的…...

MFC实现背景透明,控件不透明的对话框,且点击图片有事件响应
最终成果:背景半透明、但是控件不透明的对话框。 对话框上用图片代表功能,当点击图片时,响应点击事件,弹出相对应的对话框。 对话框固定大小,不可放大缩小,以免影响图片的显示数量。 步骤一:背景…...

案例01-tlias智能学习辅助系统01-增删改查+参数传递
目录 1、需求说明:实现对部门表和员工表的增删改查 2、环境搭建 3、部门管理 3.1 查询部门 3.2 前后端联调 3.3 删除部门 3.4 新增部门 3.5 根据ID查询数据 3.5 修改部门 总结(Controller层参数接收): 4、员工管理 4.…...

Spring之Bean的配置与实例
Spring之Bean的配置与实例 一、Bean的基础配置1. Bean基础配置【重点】配置说明代码演示运行结果 2. Bean别名配置配置说明代码演示打印结果 3. Bean作用范围配置【重点】配置说明代码演示打印结果 二、Bean的实例化1. Bean是如何创建的2. 实例化Bean的三种方式2.1 构造方法方式…...

“不保留活动”打开,导致app返回前台崩溃问题解决
问题描述 不保留活动开关打开,把app切入后台,会导致当前展示的Activity被回收,切到前台后重建。 我们有个业务场景是,Activity里面有个ViewPager2,VP里面放Fragment,Fragment的展示需要在Activity中做一些…...

解读vue3源码(3)——watch
Vue3的watch底层源码主要是通过使用Proxy对象来实现的。在Vue3中,每个组件实例都会有一个watcher实例,用于监听组件数据的变化。当组件数据发生变化时,watcher实例会触发回调函数,从而更新组件的视图。 Vue3的watch底层源码主要涉…...

优秀简历的HR视角:怎样打造一份称心如意的简历?
简历的排版应该简洁工整,注重细节。需要注意对齐和标点符号的使用,因为在排版上的细节需要下很大功夫。除此之外,下面重点讲述几点简历内容需要注意的地方。 要点1:不相关的不要写。 尤其是与应聘岗位毫不相关的实习经历&#x…...

系统集成项目管理工程师——考试重点(三)项目管理一般知识
1.项目定义: 为达到特定的目的,使用一定资源,在确定的期间内,为特定发起人提供独特的产品、服务或成果而进行的一系列相互关联的活动的集合。 2.项目目标: 成果性目标:项目产品本身 约束性目标&…...

为什么医疗保健需要MFT来帮助保护EHR文件传输
毫无疑问,医疗保健行业需要EHR技术来处理患者,设施,提供者等之间的敏感患者信息。但是,如果没有安全的MFT解决方案,您将无法安全地传输患者文件,从而使您的运营面临遭受数据泄露,尴尬࿰…...

对项目总体把控不足,项目经理应该怎么办?
公司现状:项目人员紧缺,只有两人了解此项目技术细节,其中一个不常驻现场,另一个是执行项目经理李伟。 项目经理王博是公司元老,同时负责多个项目,工作比较忙,不常驻现场,没有参加过…...

【学习笔记】CF603E Pastoral Oddities
先不考虑数据结构部分,尝试猜一下结论。 结论:一个连通块有解当且仅当连通块的度数为偶数。 然后这题要你最大边权最小。最无脑的方法就是直接上 lct \text{lct} lct。真省事啊 我第一眼想到的还是整体二分。这玩意非常好写。 但是为什么也可以用线段…...

如何使用ESP32-CAM构建一个人脸识别系统
有许多人识别系统使用签名、指纹、语音、手部几何、人脸识别等来识别人,但除了人脸识别系统。 人脸识别系统不仅可以用于安全目的来识别公共场所的人员,还可以用于办公室和学校的考勤目的。 在这个项目中,我们将使用 ESP32-CAM 构建一个人脸识…...

JavaWeb分页条件查询参数特殊字符处理
问题背景 在项目开发过程中,基本都会有列表条件查询,例如用户管理会有通过用户姓名模糊查询用户,课程管理会有课程名称模糊查询课程等等。 而查询过程中如果用户在界面上输入一些特殊字符,例如:%_等等,这…...

ubuntu18服务安装
一、JDK安装 将jdk解压缩到该目录 /opt/ sudo tar -zxvf jdk-8u261-linux-x64.tar.gz -C /opt/ #重命名 cd /opt sudo mv jdk-8u261-linux-x64 jdk_8 修改环境变量 sudo vi ~/.bashrc #在文件最后追加以下文本 #进入编辑器后输入以下指令: #1. G //将光标移到最后一…...

这些使用工具大推荐,现在知道不晚
1.Snip Snip是一款截图软件,它突出的优点就是可以制作滚动截图。 例如:对整个网页进行截图,使用Snip即可轻松获取,无需处理水印。 2.Sleep Cycle 快节奏、高压力的生活导致我们越来越晚睡觉,睡眠质量越来越差。 想提…...

【Java|golang】1048. 最长字符串链
给出一个单词数组 words ,其中每个单词都由小写英文字母组成。 如果我们可以 不改变其他字符的顺序 ,在 wordA 的任何地方添加 恰好一个 字母使其变成 wordB ,那么我们认为 wordA 是 wordB 的 前身 。 例如,“abc” 是 “abac”…...

Hive基础和使用详解
文章目录 一、启动hive1. hive启动的前置条件2. 启动方式一: hive命令3. 方式二:使用jdbc连接hive 二、Hive常用交互命令1. hive -help 命令2. hive -e 命令3. hive -f 命令4. 退出hive窗口5. 在hive窗口中执行dfs -ls /; 三、Hive语法1.DDL语句1.1 创建数据库1.2 两…...

c/c++:栈帧,传值,传址,实参传值给形参,传地址指针给形参
c/c:栈帧,传值,传址,实参传值给形参,传地址指针给形参 2022找工作是学历、能力和运气的超强结合体,遇到寒冬,大厂不招人,此时学会c的话, 我所知道的周边的会c的同学&…...

玩元宇宙血亏后 蓝色光标梭哈AI也挺悬
蓝色光标2022年年度报告出炉,巨亏21.75 亿元,其中20.38亿亏损因商誉、无形资产及其他资产减值造成,而在实际亏损业务中,元宇宙占比不小。 蓝色光标在元宇宙领域的布局,主要通过三家子公司实施,分别为蓝色宇…...

生物---英文
标题 前言必学场景词汇及用法鸟类昆虫类哺乳类爬行类情境常用单词鸟类虫类哺乳类两栖类与爬行类分类与动物相关的习语前言 加油 必学场景词汇及用法 鸟类 1bird [b[插图]d] n.鸟bird’s-eye-view[ˈb[插图]dzaɪˌvju]adj.鸟瞰图的a bird’s-eye view鸟瞰a flock of bird…...

ENVI 国产高分2号(GF-2)卫星数据辐射定标 大气校正 影像融合
1.数据 高分2号卫星数据,包含: MSS-1\2多光谱数据,4m分辨率; Pan-1\2全色波段数据,0.8m分辨率。 2.处理软件 ENVI5.3 国产插件下载地址:ENVI App Store (geoscene.cn) 首先下载插件文件; …...

操作系统考试复习——第二章 进程控制 同步与互斥
进程控制一般是由OS中的原语来实现的。 大多数OS内核都包含了两大方面的功能: 1.支撑功能:1)中断处理 2)时钟管理 3)原语操作(原语操作就是原子操作。所谓原子操作就是一个操作中所有动作要不全做要不全不做) 2.资源管理功能:1)进程管理…...

mac gitstats查看git提交记录
一、介绍: 进一步来讲,Gitstats它是一个git仓库分析软件,它可以检查仓库并生成历史数据的统计信息。可以帮助你查看git仓库的提交状态,根据不同维度分析计算,并自动生成数据图表。 官网介绍:http://gitst…...