Redis的Java客户端

 目录

1.Jedis的使用

前置工作-ssh进行端口转发

JedisAPI的使用

Jedis连接池

2.SpringDataRedis的使用

1.创建项目

2.配置文件

3.注入RedisTemplate对象

4.编写代码

3.SpringRedisTemplate

哈希结构用法

​总结


1.Jedis的使用

Jedis:以Redis命令作为方法名称,学习成本低,简单实用。但是Jedis实例 是线程不安全的,多线程环境下需要基于连接池来使用

lettuce:基于Netty实现,支持同步异步,响应式编程方式,并且是线程安全的,支持redis哨兵模式,集群模式和管道模式

前置工作-ssh进行端口转发

从本机远程访问redis服务器是需要通过云服务器的外网ip来访问linux服务器的

只修改成外网ip还不够,6379端口是被云服务器的防火墙给保护起来的,如果将该端口的防火墙

关闭,就很可能会遭到黑客攻击。每次开放一个端口,被攻击的几率就更大。

保护了之后,不开启防火墙,我们也无法访问,关闭防火墙,会遭受攻击;换成其他端口也不行,只要redis的端口被公开到公网上,就特别容易被入侵!

可以通过ssh进行端口转发

端口转发(Port Forwarding),也称为端口映射,是通过 SSH 协议在本地和远程主机之间建立一个安全的通道,将流量从一个端口转发到另一个端口的过程。

通常情况下,网络中的设备通过端口号来识别和区分不同的服务或应用程序。端口转发可以实现以下两种方式:

  1. 本地端口转发(Local Port Forwarding):在本地主机上建立一个监听端口,并将该端口上接收到的请求转发到远程主机的目标端口。这使得本地主机能够访问远程主机上的服务,就像这些服务在本地运行一样。

  2. 远程端口转发(Remote Port Forwarding):在远程主机上建立一个监听端口,并将该端口上接收到的请求转发到本地主机的目标端口。这使得远程主机能够访问本地主机上的服务,就像这些服务在远程主机运行一样。

解决方案:通过ssh端口转发,把云服务器上的redis端口,映射表到本地主机

描述:

通过windows访问云服务器的6379,访问不了,通过ssh数据报,将访问redis的请求放到ssh数据包中。通过ssh数据报来访问,服务器的ssh程序解析出上述请求,然后将数据报交给6379端口程序;

ssh也需要给多个端口传递数据,因此为了区分不同的端口,会把服务器的端口在本地用一个端口来进行表示

通过端口转发,可以绕过网络防火墙的限制,访问被阻止或限制的服务。

ssh协议是通过22端口进行的,这个端口不容易被攻破

 此时用户端程序访问127.0.0.1:8888等价于访问linux的6379端口了。同时也保障了安全性

本机查看是否将8888端口绑定了 ,LISTEN状态就是绑定了

JedisAPI的使用

上述简单配置后,我们就能连接上redis了

1.创建Maven项目,导入jedis的依赖和单元测试依赖

    <dependencies><!-- https://mvnrepository.com/artifact/redis.clients/jedis --><dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId><version>3.7.0</version></dependency><dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter</artifactId><version>5.7.2</version><scope>test</scope></dependency></dependencies>

编写代码

import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import redis.clients.jedis.Jedis;public class RedisTest {private Jedis jedis;
//前置工作,连接redis@BeforeEachvoid setUp(){jedis = new Jedis("127.0.0.1",8888);jedis.auth("123456");jedis.select(0);}
//使用@Testvoid testString1(){String res = jedis.set("hello3","hellohello");System.out.println(res);}@Testvoid testString(){String res = jedis.get("hello3");System.out.println(res);}
//后置工作,关闭资源@AfterEachvoid terDown(){if(jedis!=null){jedis.close();}}
}

 Jedis连接池

Jedis是线程不安全的,并发环境下需要创建独立的jedis对象。频繁的创建和销毁对象会有很大的开销

因此推荐使用jedis连接池代替直连方式

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;public class JedisConnectionFactory {private static final JedisPool jedispoll;static {//配置连接池JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();jedisPoolConfig.setMaxTotal(8);//最大连接数,最多常见jedisPoolConfig.setMaxIdle(8);//最大空闲连接,最多预备的jedisPoolConfig.setMinIdle(0);//最小连接,一直没人访问,就释放jedisPoolConfig.setMaxWaitMillis(1000);//等待时长,没有连接池需要等待空闲连接吗,没有等待1000ms//创建对象jedispoll = new JedisPool(jedisPoolConfig,"127.0.0.1",8888,1000,"123456");}public static Jedis getJedis(){return jedispoll.getResource();}
}

修改测试代码 

jedis = JedisConnectionFactory.getJedis();//不再new了,而是使用连接池中的jedis对象

public class RedisTest {private Jedis jedis;@BeforeEachvoid setUp(){
//        jedis = new Jedis("127.0.0.1",8888);jedis = JedisConnectionFactory.getJedis();jedis.auth("123456");jedis.select(0);}@Testvoid testString1(){String res = jedis.set("hello5","hellohello");System.out.println(res);}@Testvoid testString(){String res = jedis.get("hello5");System.out.println(res);}@AfterEachvoid terDown(){if(jedis!=null){jedis.close();}}
}

2.SpringDataRedis的使用

Spring提供了一组API,SpringDataRedis,底层做了很多兼容。包含对各种数据库的集成,对redis集成模块叫做SpringDataRedis

特点:提供了对不同Redis客户端的整合(lettuce和jedis)

提供了RedisTemplate统一API来操作Redis

支持Redis的发布订阅模型、支持哨兵和集群、支持基于letuuce的响应式编程

支持基于JDK,JSON,字符串,Spring对象的数据序列化与反序列化

支持基于Redis的JDKCOLLECTION实现 

使用 SpringDataRedis分三步:

1.创建项目,导入依赖

2.在配置文件中配置好信息

3.注入RedisTemplate对象

4.编写代码

1.创建项目

创建Maven项目,导入jedis的依赖和单元测试依赖

导入依赖,还要导入commons-pool2依赖

	<dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-pool2</artifactId></dependency></dependencies>

2.配置文件

spring.redis.host=127.0.0.1
#Redis服务器连接端口
spring.redis.port=8888
#Redis服务器连接密码(默认为空)
spring.redis.password=123456
#连接池最大连接数(使用负值表示没有限制)
spring.redis.pool.max-active=8
#连接池最大阻塞等待时间(使用负值表示没有限制)
spring.redis.pool.max-wait=-1
#连接池中的最大空闲连接
spring.redis.pool.max-idle=8
#连接池中的最小空闲连接
spring.redis.pool.min-idle=0
#连接超时时间(毫秒)
spring.redis.timeout=30000

3.注入RedisTemplate对象

	@Autowiredprivate RedisTemplate redisTemplate;

4.编写代码

@SpringBootTest
class RedisDemo3ApplicationTests {@Autowiredprivate RedisTemplate redisTemplate;@Testvoid contextLoads() {redisTemplate.opsForValue().set("name","zhangsan");Object name = redisTemplate.opsForValue().get("name");System.out.println(name);}}

运行

redis中的还没有修改 ,查询后发现乱码了

 这与我们之前谈到的SpringDataRedis集成的redis支持序列化的特性

客户端传输数据时,将Object对象序列化为字节形式,默认采用的是jdk序列化,得到的结果自然不是正常字符串,可读性差并且占用内存

看一段RedisTemplate类的源码

public class RedisTemplate<K, V> extends RedisAccessor implements RedisOperations<K, V>, BeanClassLoaderAware {private boolean enableTransactionSupport = false;private boolean exposeConnection = false;private boolean initialized = false;private boolean enableDefaultSerializer = true;@Nullableprivate RedisSerializer<?> defaultSerializer;@Nullableprivate ClassLoader classLoader;@Nullableprivate RedisSerializer keySerializer = null;@Nullableprivate RedisSerializer valueSerializer = null;@Nullableprivate RedisSerializer hashKeySerializer = null;@Nullableprivate RedisSerializer hashValueSerializer = null;private RedisSerializer<String> stringSerializer = RedisSerializer.string();/省略......./public void afterPropertiesSet() {super.afterPropertiesSet();boolean defaultUsed = false;if (this.defaultSerializer == null) {this.defaultSerializer = new JdkSerializationRedisSerializer(this.classLoader != null ? this.classLoader : this.getClass().getClassLoader());}

提供了各种数据结构的序列化,并且最后一个方法,当默认序列化设置为空时,首先会创建一个新的JdkSerializationRedisSerializer。这个不太好用,还有其他的序列化方式

查看RedisSerializer的层级结构

我们可以自定义 RedisTemplate的序列化 方式

值处理需要使用json,需要引入依赖

		<dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId></dependency>

编写序列化配置代码

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 {@Beanpublic RedisTemplate<String,Object> redisTemplate(RedisConnectionFactory connectionFactory){//创建RedisTemplate对象RedisTemplate<String,Object> redisTemplate = new RedisTemplate<>();//设置连接工厂redisTemplate.setConnectionFactory(connectionFactory);//创建JSON序列化工具GenericJackson2JsonRedisSerializer jsonRedisSerializer = new GenericJackson2JsonRedisSerializer();//设置key的序列化redisTemplate.setKeySerializer(RedisSerializer.string());redisTemplate.setHashKeySerializer(RedisSerializer.string());//设置value序列化redisTemplate.setValueSerializer(jsonRedisSerializer);redisTemplate.setHashValueSerializer(jsonRedisSerializer);//返回return redisTemplate;}
}
@SpringBootTest
class RedisDemo3ApplicationTests {@Autowiredprivate RedisTemplate<String,Object> redisTemplate;@Testvoid contextLoads() {redisTemplate.opsForValue().set("name","慧怡");Object name = redisTemplate.opsForValue().get("name");System.out.println(name);}
}

我们使用Redis Desktop Manager查看

Redis Desktop Manager是一款简单快速、跨平台的Redis桌面管理工具,也被称作Redis可视化工具;支持命令控制台操作,以及常用,查询key,rename,delete等操作。

再看是否能传输对象

	@Data@NoArgsConstructor@AllArgsConstructorpublic class User {private String name;private Integer age;}@Testvoid testSaveUser(){//写入数据redisTemplate.opsForValue().set("user:100",new User("慧怡",21));//获取数据User o = (User)redisTemplate.opsForValue().get("user:100");System.out.println("o = "+o);}

 

 

 可以看到:将对象序列化后,存入了json的对象,获取结果时,控制台打印的时反序列化了的对象

3.SpringRedisTemplate

写入json对象的同时,还写入了一个"@class": "com.example.demo.poji.User",所以在反序列化的时候,才根据这个读取到字节码,类的名称,将json反序列化为User对象

所以无论是字符串,还是java对象,都可以进行存储了,我们只需要在config中配置好 key的序列化工具和 value 的序列化工具就好

同时也带来一个问题

可以看到第一行数据比下面的数据都长了,会带来额外的内存开销,如果数据量巨大,将是一笔不小的开销,但是为了能反序列化又不能不传入该字段

为了节省内存空间,并不会使用JSON序列化器处理VALUE,而是统一使用String序列化器,要求只存储String类型的KEY和VALUE,当存储Java对象时,手动完成对象的序列化和反序列化

转换过程: 

那么是否要重新将配置文件的序列化改动呢?是不用的, SpringRedisTemplate类的KEY与VALUE默认就是String方式,省去了我们自定义的过程

@SpringBootTest
public class RedisDemo3StringTests {@Autowiredprivate StringRedisTemplate stringRedisTemplate;@Testvoid testString(){stringRedisTemplate.opsForValue().set("name","慧怡");Object name = stringRedisTemplate.opsForValue().get("name");System.out.println(name);}

 

 传对象时:

1.创建对象并手动序列化->写入数据

2.获取数据->手动反序列化

    @Autowiredprivate static final ObjectMapper mapper = new ObjectMapper();@Testvoid testSaveUser() throws JsonProcessingException {//创建对象User user = new User("慧怡",21);//手动序列化String json = mapper.writeValueAsString(user);//写入数据stringRedisTemplate.opsForValue().set("user:200",json);//获取数据String jsonUser = stringRedisTemplate.opsForValue().get("user:200");//手动反序列化User user1 = mapper.readValue(jsonUser,User.class);System.out.println("user1 = "+user1);}

哈希结构用法

    @Testvoid hsahTest(){//存储stringRedisTemplate.opsForHash().put("user:300","name","慧怡");stringRedisTemplate.opsForHash().put("user:300","age","21");//取值Map<Object,Object> entires = stringRedisTemplate.opsForHash().entries("user:300");System.out.println(entires);}

 

总结

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

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

相关文章

蓝桥杯,我劝你不要参加的8个完美理由

蓝桥杯&#xff0c;是一个全国高校的IT技术比拼&#xff0c;如果你参加了&#xff0c;可能不止是刷题数量的剧增&#xff0c;还有你的软件人生 我劝你不要参加&#xff0c;因为如果你参加了&#xff0c;可能会有以下烦恼&#xff1a; 目录 1、会让你变得上进 2、会提前感受码…

windows下mysql的下载与安装

文章目录 1 下载2 安装目录下新建data文件夹和my.ini3 安装4设置密码与远程连接5 配置环境变量6 navicate连接成功 1 下载 官网地址 https://www.mysql.com/点击下载 社区下载 社区服务 选择版本下载 2 安装目录下新建data文件夹和my.ini my.ini 内容如下 [mysql] # 设置my…

算法与数据结构(五)--树【1】树与二叉树是什么

一.树的定义 树是一个具有层次结构的集合&#xff0c;是由一个有限集和集合上定义的一种层次结构关系构成的。不同于线性表&#xff0c;树并不是线性的&#xff0c;而是有分支的。 树&#xff08;Tree&#xff09;是n&#xff08;n>0&#xff09;个结点的有限集。 若n0&…

ChatGPT能否撰写科研论文?

ChatGPT&#xff0c;这款被许多人誉为语言处理领域的“黑马”&#xff0c;究竟能否应用于撰写科研论文&#xff1f;近期&#xff0c;以色列理工学院生物学家兼数据科学家Roy Kishony带领的团队&#xff0c;针对这一问题进行了系列研究&#xff0c;其结果已在《Nature》杂志上发…

正则匹配img标签里面src

正则&#xff1a; (?<src\s*\s*\")\S(?\"{1})匹配效果&#xff1a;

golang waitgroup

案例 WaitGroup 可以解决一个 goroutine 等待多个 goroutine 同时结束的场景&#xff0c;这个比较常见的场景就是例如 后端 worker 启动了多个消费者干活&#xff0c;还有爬虫并发爬取数据&#xff0c;多线程下载等等。 我们这里模拟一个 worker 的例子 package mainimport (…

LLM - LoRA 模型合并与保存

目录 一.引言 二.LoRA 1.LoRA 简介 2.LoRA 参数 3.LoRA 合并 4.LoRA 保存 三.总结 一.引言 LLM 使用过程中最常用方法之一就是通过 LoRA 基于自己的数据对大模型进行微调&#xff0c;本文简单介绍 LoRA 原理以及如何合并多个 LoRA 模型并保存。 peft0.4.0 transforme…

[NLP]LLM高效微调(PEFT)--LoRA

LoRA 背景 神经网络包含很多全连接层&#xff0c;其借助于矩阵乘法得以实现&#xff0c;然而&#xff0c;很多全连接层的权重矩阵都是满秩的。当针对特定任务进行微调后&#xff0c;模型中权重矩阵其实具有很低的本征秩&#xff08;intrinsic rank&#xff09;&#xff0c;因…

记一次centos 磁盘挂载过程

前言 最近买了云服务器磁盘&#xff0c;需要挂载&#xff0c;一下就由大猿来记录这次过程。 挂载过程 查看磁盘挂载情况 查看物理硬盘 lsblkfdisk -l标记分区 fdisk /dev/vdb格式化分区 xfs mkfs.xfs /dev/vdb mkfs.xfs -f /dev/vdbext4 mkfs.ext4 /dev/vdbxfs 和 ex…

HTTP——HTTP报文内的HTTP信息

HTTP 通信过程包括从客户端发往服务器端的请求及从服务器端返回客户端的响应。本章就让我们来了解一下请求和响应是怎样运作的。 HTTP 一、HTTP报文二、请求报文及响应报文的结构三、编码提升传输速率1、报文主体和实体主题的差异2、压缩传输的内容编码3、分割发送的分块传输编…

Spring MVC异步上传、跨服务器上传和文件下载

一、异步上传 之前的上传方案&#xff0c;在上传成功后都会跳转页面。而在实际开发中&#xff0c;很多情况下上传后不进行跳转&#xff0c;而是进行页面的局部刷新&#xff0c;比如&#xff1a;上传头像成功后将头像显示在网页中。这时候就需要使用异步文件上传。 1.1 JSP页面 …

为了规避风险,如何给大模型打水印?

大型语言模型&#xff0c;如最近开发的ChatGPT&#xff0c;可以撰写文件、创建可执行代码和回答问题&#xff0c;通常具有人类般的能力。 随着这些大模型的应用越来越普遍&#xff0c;越来越大的风险也显现了出来&#xff0c;它们可能被用于恶意目的。这些恶意目的包括&#xf…

【漏洞复现】Metabase 远程命令执行漏洞(CVE-2023-38646)

文章目录 前言声明一、漏洞介绍二、影响版本三、漏洞原理四、漏洞复现五、修复建议 前言 Metabase 0.46.6.1之前版本和Metabase Enterprise 1.46.6.1之前版本存在安全漏洞&#xff0c;未经身份认证的远程攻击者利用该漏洞可以在服务器上以运行 Metabase 服务器的权限执行任意命…

虹科活动 | 走进宇通客车-汽车新供应链技术展精彩回顾

引言 7月27日&#xff0c;走进宇通客车-汽车新供应链技术展于宇通研发中心成功举办&#xff0c;本次展会中虹科为大家带来了一体化车载天线与车辆GNSS仿真测试方案&#xff0c;感谢您前来探讨与交流&#xff01; 精彩产品一览 车辆GNSS仿真测试方案 虹科高性能GNSS模拟器具有灵…

如何搭建WordPress博客网站,并且发布至公网上?

如何搭建WordPress博客网站&#xff0c;并且发布至公网上&#xff1f; 文章目录 如何搭建WordPress博客网站&#xff0c;并且发布至公网上&#xff1f;概述前置准备1 安装数据库管理工具1.1 安装图形图数据库管理工具&#xff0c;SQL_Front 2 创建一个新数据库2.1 创建数据库2.…

振弦传感器信号转换器应用山体滑坡安全监测

振弦传感器信号转换器应用山体滑坡安全监测 随着人类文明的进步&#xff0c;自然灾害对人们的生活和财产安全造成的威胁也越来越大。山体滑坡作为自然灾害中的一种&#xff0c;给人们的生活和财产安全带来了极大的威胁。因此&#xff0c;进行山体滑坡的安全监测显得尤为重要。振…

Chat模块封装

封装保存用户类 utils/chat.js class Chat{constructor(){// 当前登录的用户this._user null;// 会话数组 和多个人this._sessions []; //user message// 当前会话 &#xff08;和谁在聊天&#xff09;this._current_session null;}setUser(user){this._user user} }exp…

C++中数据的输入输出介绍

C中数据的输入输出介绍 C中数据的输入输出涉及到的文件 <iostream>&#xff1a;这是C标准库中最常用的头文件之一&#xff0c;包含了进行标准输入输出操作的类和对象&#xff0c;如std::cin、std::cout、std::endl等。 <iomanip>&#xff1a;该头文件提供了一些用…

免费商用 Meta 发布开源大语言模型 Llama 2

Meta 和微软深度合作&#xff0c;正式推出下一代开源大语言模型 Llama 2&#xff0c;并宣布免费提供给研究和商业使用。 Llama 2 论文地址&#xff1a;Llama 2: Open Foundation and Fine-Tuned Chat Models 据介绍&#xff0c;相比于 Llama 1&#xff0c;Llama 2 的训练数据多…

AD21 PCB设计的高级应用(九)3D PDF的输出

&#xff08;九&#xff09;3D PDF的输出 1.3D PDF的输出2.制作PCB 3D视频 1.3D PDF的输出 Altium Designer 19 带有 3D输出功能,能够直接将 PCB 的 3D效果输出到 PDF 中。 ’(1)打开带有 3D 模型的 PCB 文件,执行菜单栏中“文件”→“导出”→“PDF3D”命令&#xff0c;选择…