Il est nécessaire d'interroger le produit en utilisant le nom du produit. Cela nécessite d'abord d'interroger le cache, et s'il est introuvable, accédez à la base de données pour l'interroger. ; après avoir interrogé la base de données, ajoutez-le au cache, puis lors de l'interrogation, continuez d'abord à interroger le cache.
Vous pouvez rédiger un jugement conditionnel, le pseudo code est le suivant :
//先从缓存中查询 String goodsInfoStr = redis.get(goodsName); if(StringUtils.isBlank(goodsInfoStr)){ //如果缓存中查询为空,则去数据库中查询 Goods goods = goodsMapper.queryByName(goodsName); //将查询到的数据存入缓存 goodsName.set(goodsName,JSONObject.toJSONString(goods)); //返回商品数据 return goods; }else{ //将查询到的str转换为对象并返回 return JSON.parseObject(goodsInfoStr, Goods.class); }
pas réutilisable
et ne peut être utilisée que dans ce scénario. En supposant qu'il existe de nombreux besoins similaires à la requête de produit ci-dessus dans notre système, nous devons alors écrire un tel
if(...)else{...}
partout. En tant que programmeur, il est très inconfortable de ne pas pouvoir unifier des codes similaires ou répétés. Le code de ce scénario doit donc être optimisé.
不可复用的
,只能用在这个场景。假设在我们的系统中还有很多类似上面商品查询的需求,那么我们需要到处写这样的if(...)else{...}
。作为一个程序员,不能把类似的或者重复的代码统一起来是一件很难受的事情,所以需要对这种场景的代码进行优化。
上面这串代码的问题在于:入参不固定、返回值也不固定,如果仅仅是参数不固定,使用泛型即可。但最关键的是查询方法也是不固定的,比如查询商品和查询用户肯定不是一个查询方法吧。
所以如果我们可以把一个方法(即上面的各种查询方法)也能当做一个参数传入一个统一的判断方法就好了,类似于:
/** * 这个方法的作用是:先执行method1方法,如果method1查询或执行不成功,再执行method2方法 */ public staticT selectCacheByTemplate(method1,method2)
想要实现上面的这种效果,就不得不提到Java8的新特性:函数式编程
在Java中有一个package:java.util.function
,里面全部是接口,并且都被@FunctionalInterface
Les paramètres d'entrée ne sont pas corrigés, et la valeur de retour n'est pas corrigée non plus. Si seuls les paramètres ne sont pas corrigés, utilisez des génériques. . Mais le plus important est que la méthode de requête n'est pas fixe. Par exemple, interroger des produits et interroger des utilisateurs n'est certainement pas la même méthode de requête.Ce serait donc génial si nous pouvions passer une méthode (c'est-à-dire les différentes méthodes de requête ci-dessus) en tant que paramètre dans une méthode de jugement unifiée, similaire à :
4.0.0 org.springframework.boot spring-boot-starter-parent 2.7.2 com.example SpringBoot-query 0.0.1-SNAPSHOT SpringBoot-query Demo project for Spring Boot 1.8 org.springframework.boot spring-boot-starter-data-redis com.alibaba fastjson 1.2.83 org.springframework.boot spring-boot-starter-web org.springframework.boot spring-boot-starter-test test org.springframework.boot spring-boot-maven-plugin
Programmation fonctionnelle
# #Il y a un package en Java :java.util.function
, qui contient toutes les interfaces, et elles sont toutes modifiées par l'annotation@FunctionalInterface
.
Accepter les paramètres, Aucune valeur de retourOpérateur :accepte les paramètres et renvoie des valeurs du même type que les paramètres
# #Predicate ):Accepte les paramètres et renvoie le type booléen
Fournisseur (supply):Aucun paramètre, valeur de retour # #
Code du projet
package com.example.springbootquery; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication public class SpringBootQueryApplication { public static void main(String[] args) { SpringApplication.run(SpringBootQueryApplication.class, args); } }
Parmi eux, CacheService interroge les données du cache et GoodsService interroge les données de la base de données
SpringBootQueryApplication . javapackage com.example.springbootquery.entity; public class Goods { private String goodsName; private Integer goodsTotal; private Double price; public String getGoodsName() { return goodsName; } public void setGoodsName(String goodsName) { this.goodsName = goodsName; } public Integer getGoodsTotal() { return goodsTotal; } public void setGoodsTotal(Integer goodsTotal) { this.goodsTotal = goodsTotal; } public Double getPrice() { return price; } public void setPrice(Double price) { this.price = price; } @Override public String toString() { return "Goods{" + "goodsName='" + goodsName + '\'' + ", goodsTotal='" + goodsTotal + '\'' + ", price=" + price + '}'; } }
package com.example.springbootquery.function; @FunctionalInterface public interface CacheSelector{ T select() throws Exception; }
package com.example.springbootquery.service; import com.example.springbootquery.entity.Goods; public interface CacheService { /** * 从缓存中获取商品 * * @param goodsName 商品名称 * @return goods */ Goods getGoodsByName(String goodsName) throws Exception; }
CacheService.java
package com.example.springbootquery.service.impl; import com.alibaba.fastjson.JSON; import com.example.springbootquery.entity.Goods; import com.example.springbootquery.service.CacheService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.stereotype.Service; @Service("cacheService") public class CacheServiceImpl implements CacheService { @Autowired private StringRedisTemplate redisTemplate; @Override public Goods getGoodsByName(String goodsName) throws Exception { String s = redisTemplate.opsForValue().get(goodsName); return null == s ? null : JSON.parseObject(s, Goods.class); } }
package com.example.springbootquery.service; import com.example.springbootquery.entity.Goods; public interface GoodsService { Goods getGoodsByName(String goodsName); }
package com.example.springbootquery.service.impl; import com.alibaba.fastjson.JSONObject; import com.example.springbootquery.entity.Goods; import com.example.springbootquery.service.GoodsService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.stereotype.Service; @Service public class GoodsServiceImpl implements GoodsService { @Autowired private StringRedisTemplate stringRedisTemplate; @Override public Goods getGoodsByName(String goodsName) { Goods goods = new Goods(); goods.setGoodsName("商品名1"); goods.setGoodsTotal(20); goods.setPrice(30.0D); stringRedisTemplate.opsForValue().set(goodsName, JSONObject.toJSONString(goods)); return goods; } }
package com.example.springbootquery.util; import com.example.springbootquery.function.CacheSelector; import java.util.function.Supplier; public class BaseUtil { /** * 缓存查询模板 * * @param cacheSelector 查询缓存的方法 * @param databaseSelector 数据库查询方法 * @return T */ public staticT selectCacheByTemplate(CacheSelector cacheSelector, Supplier databaseSelector) { try { System.out.println("query data from redis ······"); // 先查 Redis缓存 T t = cacheSelector.select(); if (t == null) { // 没有记录再查询数据库 System.err.println("redis 中没有查询到"); System.out.println("query data from database ······"); return databaseSelector.get(); } else { return t; } } catch (Exception e) { // 缓存查询出错,则去数据库查询 e.printStackTrace(); System.err.println("redis 查询出错"); System.out.println("query data from database ······"); return databaseSelector.get(); } } }
# # Parce que je ne me soucie pas des paramètres et que j'ai seulement besoin d'une valeur de retour, Supplier est utilisé ici.Usagepackage com.example.springbootquery; import com.example.springbootquery.entity.Goods; import com.example.springbootquery.service.CacheService; import com.example.springbootquery.service.GoodsService; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import static com.example.springbootquery.util.BaseUtil.selectCacheByTemplate; @SpringBootTest class SpringBootQueryApplicationTests { @Autowired private CacheService cacheService; @Autowired private GoodsService userService; @Test void contextLoads() throws Exception { Goods user = selectCacheByTemplate( () -> cacheService.getGoodsByName("商品名1"), () -> userService.getGoodsByName("商品名1") ); System.out.println(user); } }Copier après la connexionrrreee
Première requête à partir des donnéesRequête à partir du cache pour la deuxième fois
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!