Redis在實際專案中的應用
阿新 • • 發佈:2019-02-07
想起自己以前想把Redis整合到JavaWeb專案中,網上搜了很多教程都不全面,現在我終於弄明白了,所以想在這分享一下。
1.Redis的安裝
我這裡就不講了,網上有很多教程,windows,Linux,我自己的是搭建在伺服器上的。圖形化連線
向業務邏輯中新增快取
1.1. 介面封裝
常用的操作redis的方法提取出一個介面,分別對應單機版和叢集版建立兩個實現類。
1.1.1. 介面定義
##jedisClient
package cn.e3mall.common.jedis; import java.util.List; public interface JedisClient { String set(String key, String value); String get(String key); Boolean exists(String key); Long expire(String key, int seconds); Long ttl(String key); Long incr(String key); Long hset(String key, String field, String value); String hget(String key, String field); Long hdel(String key, String... field); Boolean hexists(String key, String field); List<String> hvals(String key); Long del(String key); }
###JedisClientPool
package cn.e3mall.common.jedis; import java.util.List; import redis.clients.jedis.Jedis; import redis.clients.jedis.JedisPool; public class JedisClientPool implements JedisClient { private JedisPool jedisPool; public JedisPool getJedisPool() { return jedisPool; } public void setJedisPool(JedisPool jedisPool) { this.jedisPool = jedisPool; } @Override public String set(String key, String value) { Jedis jedis = jedisPool.getResource(); String result = jedis.set(key, value); jedis.close(); return result; } @Override public String get(String key) { Jedis jedis = jedisPool.getResource(); String result = jedis.get(key); jedis.close(); return result; } @Override public Boolean exists(String key) { Jedis jedis = jedisPool.getResource(); Boolean result = jedis.exists(key); jedis.close(); return result; } @Override public Long expire(String key, int seconds) { Jedis jedis = jedisPool.getResource(); Long result = jedis.expire(key, seconds); jedis.close(); return result; } @Override public Long ttl(String key) { Jedis jedis = jedisPool.getResource(); Long result = jedis.ttl(key); jedis.close(); return result; } @Override public Long incr(String key) { Jedis jedis = jedisPool.getResource(); Long result = jedis.incr(key); jedis.close(); return result; } @Override public Long hset(String key, String field, String value) { Jedis jedis = jedisPool.getResource(); Long result = jedis.hset(key, field, value); jedis.close(); return result; } @Override public String hget(String key, String field) { Jedis jedis = jedisPool.getResource(); String result = jedis.hget(key, field); jedis.close(); return result; } @Override public Long hdel(String key, String... field) { Jedis jedis = jedisPool.getResource(); Long result = jedis.hdel(key, field); jedis.close(); return result; } @Override public Boolean hexists(String key, String field) { Jedis jedis = jedisPool.getResource(); Boolean result = jedis.hexists(key, field); jedis.close(); return result; } @Override public List<String> hvals(String key) { Jedis jedis = jedisPool.getResource(); List<String> result = jedis.hvals(key); jedis.close(); return result; } @Override public Long del(String key) { Jedis jedis = jedisPool.getResource(); Long result = jedis.del(key); jedis.close(); return result; } }
###JedisClientCluster
package cn.e3mall.common.jedis; import java.util.List; import redis.clients.jedis.JedisCluster; public class JedisClientCluster implements JedisClient { private JedisCluster jedisCluster; public JedisCluster getJedisCluster() { return jedisCluster; } public void setJedisCluster(JedisCluster jedisCluster) { this.jedisCluster = jedisCluster; } @Override public String set(String key, String value) { return jedisCluster.set(key, value); } @Override public String get(String key) { return jedisCluster.get(key); } @Override public Boolean exists(String key) { return jedisCluster.exists(key); } @Override public Long expire(String key, int seconds) { return jedisCluster.expire(key, seconds); } @Override public Long ttl(String key) { return jedisCluster.ttl(key); } @Override public Long incr(String key) { return jedisCluster.incr(key); } @Override public Long hset(String key, String field, String value) { return jedisCluster.hset(key, field, value); } @Override public String hget(String key, String field) { return jedisCluster.hget(key, field); } @Override public Long hdel(String key, String... field) { return jedisCluster.hdel(key, field); } @Override public Boolean hexists(String key, String field) { return jedisCluster.hexists(key, field); } @Override public List<String> hvals(String key) { return jedisCluster.hvals(key); } @Override public Long del(String key) { return jedisCluster.del(key); } }
配置:applicationContext-redis.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:context="http://www.springframework.org/schema/context" xmlns:p="http://www.springframework.org/schema/p"
xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.2.xsd
http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-4.2.xsd">
<!-- 連線redis單機版 -->
<bean id="jedisClientPool" class="cn.e3mall.common.jedis.JedisClientPool">
<property name="jedisPool" ref="jedisPool"></property>
</bean>
<bean id="jedisPool" class="redis.clients.jedis.JedisPool">
<constructor-arg name="host" value="115.159.126.205"/>
<constructor-arg name="port" value="6379"/>
</bean>
<!-- 連線redis叢集 -->
<!-- <bean id="jedisClientCluster" class="cn.e3mall.common.jedis.JedisClientCluster">
<property name="jedisCluster" ref="jedisCluster"/>
</bean>
<bean id="jedisCluster" class="redis.clients.jedis.JedisCluster">
<constructor-arg name="nodes">
<set>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="192.168.25.162"></constructor-arg>
<constructor-arg name="port" value="7001"></constructor-arg>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="192.168.25.162"></constructor-arg>
<constructor-arg name="port" value="7002"></constructor-arg>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="192.168.25.162"></constructor-arg>
<constructor-arg name="port" value="7003"></constructor-arg>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="192.168.25.162"></constructor-arg>
<constructor-arg name="port" value="7004"></constructor-arg>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="192.168.25.162"></constructor-arg>
<constructor-arg name="port" value="7005"></constructor-arg>
</bean>
<bean class="redis.clients.jedis.HostAndPort">
<constructor-arg name="host" value="192.168.25.162"></constructor-arg>
<constructor-arg name="port" value="7006"></constructor-arg>
</bean>
</set>
</constructor-arg>
</bean> -->
</beans>
封裝程式碼測試
@Test
public void testJedisClient() throws Exception {
//初始化Spring容器
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring/applicationContext-redis.xml");
//從容器中獲得JedisClient物件
JedisClient jedisClient = applicationContext.getBean(JedisClient.class);
jedisClient.set("first", "100");
String result = jedisClient.get("first");
System.out.println(result);
}
新增快取
1.1.1. 功能分析
查詢內容列表時新增快取。
1、查詢資料庫之前先查詢快取。
2、查詢到結果,直接響應結果。
3、查詢不到,快取中沒有需要查詢資料庫。
4、把查詢結果新增到快取中。
5、返回結果。
向redis中新增快取:
Key:cid
Value:內容列表。需要把java物件轉換成json。
使用hash對key進行歸類。
HASH_KEY:HASH
|--KEY:VALUE
|--KEY:VALUE
|--KEY:VALUE
|--KEY:VALUE
注意:新增快取不能影響正常業務邏輯。
程式碼實現(在服務層實現)
@Autowired
private JedisClient jedisClient;
@Value("${CONTENT_LIST}")
private String CONTENT_LIST;
@Override
public E3Result addContent(TbContent content) {
//將內容資料插入到內容表
content.setCreated(new Date());
content.setUpdated(new Date());
//插入到資料庫
contentMapper.insert(content);
//快取同步,刪除快取中對應的資料。
jedisClient.hdel(CONTENT_LIST, content.getCategoryId().toString());
return E3Result.ok();
}
@Override
public List<TbContent> getContentListByCid(long cid) {
//查詢快取
try {
//如果快取中有直接響應結果
String json = jedisClient.hget(CONTENT_LIST, cid + "");
if (StringUtils.isNotBlank(json)) {
List<TbContent> list = JsonUtils.jsonToList(json,TbContent.class);
System.out.println("從快取中查出的資料");
return list;
}
} catch (Exception e) {
e.printStackTrace();
}
//如果沒有查詢資料庫
TbContentExample example = new TbContentExample();
Criteria criteria = example.createCriteria();
//設定查詢條件
criteria.andCategoryIdEqualTo(cid);
//執行查詢
List<TbContent> list = contentMapper.selectByExampleWithBLOBs(example);
//把結果新增到快取
try {
System.out.println("把結果新增到快取");
jedisClient.hset(CONTENT_LIST, cid + "", JsonUtils.objectToJson(list));
} catch (Exception e) {
e.printStackTrace();
}
return list;
}
**注意要實現快取同步
再講一個東西,熱快取,給它設一個過期時間(不同步也是可以的)
@Override
public TbItem getItemById(long itemId) {
//獲取商品新增快取,不影業務響邏輯,try-catch
try {
System.out.println("快取獲取商品資訊");
String json = jedisClient.get(REDIS_ITEM_PRE+":"+itemId+":BASE");
if(StringUtils.isNotBlank(json)) {
TbItem tbItem = JsonUtils.jsonToPojo(json,TbItem.class);
return tbItem;
}
}catch(Exception e) {
e.printStackTrace();
}
//快取中沒有,查詢資料庫
//根據主鍵查詢
//TbItem tbItem = itemMapper.selectByPrimaryKey(itemId);
TbItemExample example = new TbItemExample();
Criteria criteria = example.createCriteria();
//設定查詢條件
criteria.andIdEqualTo(itemId);
//執行查詢
List<TbItem> list = itemMapper.selectByExample(example);
if (list != null && list.size() > 0) {
//結果新增到快取
try {
System.out.println("快取新增商品資訊");
jedisClient.set(REDIS_ITEM_PRE+":"+itemId+":BASE",JsonUtils.objectToJson(list.get(0)));
//設定過期時間(1個小時)
jedisClient.expire(REDIS_ITEM_PRE+":"+itemId+":BASE",TIEM_CACHE_EXPIRE);
}catch(Exception e) {
e.printStackTrace();
}
return list.get(0);
}
return null;
}
redis中的資料:
{"itemId":1231490,"created":1425821627000,"updated":1425821627000,"itemDesc":"<table cellpadding=\"0\" cellspacing=\"0\" align=\"center\" border=\"0\" width=\"750\"> <tbody> <tr> <td height=\"51\" width=\"750\"><div style=\"PADDING-BOTTOM: 15px; LINE-HEIGHT: 1.5em; PADDING-LEFT: 15px; PADDING-RIGHT: 15px; FONT-FAMILY: '微軟雅黑'; FONT-SIZE: 15px; PADDING-TOP: 15px\"><span style=\"color:#ff6666;\"><strong>溫馨提示:</strong></span><strong>小米4聯通版</strong>內建運營商軟體,<strong>小米4官方聯通版</strong>是無內建運營商軟體的。</div></td> </tr> </tbody></table><div class=\"content_tpl\"> <div class=\"formwork_bt\" id=\"detail-tag-id-0\" name=\"detail-tag-id-0\" text=\"產品展示\"> <div class=\"formwork_bt_dz\"> <span>產品展示</span> <span class=\"s2\">Products Exhibition</span> </div> </div> <table cellpadding=\"0\" cellspacing=\"6\" align=\"center\" border=\"0\" width=\"750\"> <tbody><tr> <td><img src=\"http://img30.360buyimg.com/jgsq-productsoa/jfs/t346/347/1422821621/45195/e22eac9b/5439038bNa168d543.jpg\" alt=\"\" /></td> <td><p class=\"formwork_titleleft\">小米手機4</p><p class=\"formwork_titleleft2\">依然出色效能用智慧手機閱讀、拍照、遊戲甚至完成複雜艱鉅的任務,這一切都依賴於快。每一代小米手機,都使用了當前先進的元器件。更快的處理器、更快的快閃記憶體、更快的相機和影象處理,支援更快的網路。集世界全新科技在一部5英寸大小的裝置,只為讓科技樂趣人人都可享用,生活正因此變得更加美好</p></td> </tr></tbody></table> <div class=\"formwork_bt\" id=\"detail-tag-id-2\" name=\"detail-tag-id-2\" text=\"產品資訊\"> <div class=\"formwork_bt_dz\"> <span>產品資訊</span> <span class=\"s2\">Product Information</span> </div> </div> <div class=\"formwork\"><div class=\"formwork_titlecenter\">Qualcomm® 驍龍? 801四核2.5GHz 處理器</div><div class=\"formwork_text\">高階手機處理器效能之王小米手機 4 採用了高通動力十足的驍龍801手機處理器,內含四個Krait 400 2.5GHz 處理核心。運算速度提升14%,效能更強大。它能出色地同時處理多個複雜任務。它的強大還體現在影象處理器速度較前代提升近一倍,這讓拍照與錄影都有了更多玩法和可能性。內含一個 Hexagon DSP 核心,專門以超低功耗執行電影、音樂、拍照等任務。這意味著在效能更強大的同時,手機續航都比以往更加持久耐用。</div></div><div class=\"formwork\"><div class=\"formwork_img\"><img src=\"http://img20.360buyimg.com/vc/jfs/t307/358/1403468763/249668/b4048162/54388804Na68f8ead.png\" alt=\"\" /></div></div><div class=\"formwork\"><div class=\"formwork_titlecenter\">Adreno 330 圖形處理器</div><div class=\"formwork_text\">遊戲機品質的3D渲染能力Adreno 330 圖形處理器支援高階圖形處理API,包括OpenGL ES 3.0、OpenCL、RenderscriptCompute和FlexRender 。它之所以能夠快速渲染複雜的圖形,得益於統一渲染架構及FlexRender 快速渲染技術。統一渲染架構能夠根據所渲染的圖形型別,動態調整其資源分配方式,畫素和頂點渲染均可進行獨立調節。FlexRender 技術可以直接或通過延遲渲染模式在圖形畫素間動態轉換,從而幫助Adreno GPU更快速、更高效地渲染遊戲圖形。</div></div><div class=\"formwork\"><div class=\"formwork_img\"><img src=\"http://img20.360buyimg.com/vc/jfs/t352/350/1359939860/199186/e31d4b62/5438882fN8a00b5ac.png\" alt=\"\" /></div></div><div class=\"formwork\"><div class=\"formwork_titlecenter\">高色飽和度屏,84% NTSC色域</div><div class=\"formwork_text\">鮮豔度比 iPhone 5s 高17%小米手機4 採用了高色彩飽和度夏普/JDI螢幕,整體的色彩飽和度提升17%,令色彩表現力更加豐富。無論在小米手機4上面瀏覽圖片、觀看視訊,或是使用精彩的App,都能為你呈現逼真的顏色效果。</div></div><div class=\"formwork\"><div class=\"formwork_img\"><img src=\"http://img20.360buyimg.com/vc/jfs/t292/303/1377649814/236958/cf633089/54388856N6b7e3293.png\" alt=\"\" /></div></div><table cellpadding=\"0\" cellspacing=\"6\" align=\"center\" border=\"0\" width=\"750\"> <tbody><tr> <td><img src=\"http://img20.360buyimg.com/vc/jfs/t337/149/1399837173/112733/5916cfcd/54388886Neddc8729.png\" alt=\"\" /></td> <td><p class=\"formwork_titleleft\">索尼高畫質</p><p class=\"formwork_titleleft2\">最快0.3秒極速對焦如何用手機拍下轉瞬即逝的美好瞬間?採用索尼 IMX 214 第二代 1300 萬畫素 Exmor RS? 堆疊式影象感測器,通過減少片上微透鏡與感光二極體的距離,使其更容易集合光線。支援硬體更豐富、噪點更少的照片。6片鏡頭組採用了閉環式對焦技術,對焦最快至0.3秒,這比主流手機都快了兩倍之多。它還具備手機上最大的F1.8大光圈,暗光更出色,背景虛化效果更加柔美。為了讓拍攝具備更多便利和趣味,還可以使用魔術對焦功能,先拍照,回看時再選擇焦點。不必苦練攝影技能,卻可以用小米手機4拍出動人有趣的照片</p></td> </tr></tbody></table><div class=\"formwork\"><div class=\"formwork_titlecenter\">支援最新4G LTE網路</div><div class=\"formwork_text\">用更快的方式上網、聽音樂、看視訊移動4G版支援最新中國移動4G(TDD-LTE)網路,下行峰值速率最高可達到132Mbps,上傳的速度可達31Mbps。如此之快的傳輸速度幾乎可滿足對於無線應用的任何需要。當你外出旅行時,你可以用它隨時隨地觀看線上高清視訊,瀏覽網頁或玩最流行的線上遊戲,極速的4G網路速度可以帶給你酣暢淋漓的使用體驗。另可選購聯通3G及電信3G版本。</div></div><div class=\"formwork\"><div class=\"formwork_img\"><img src=\"http://img20.360buyimg.com/vc/jfs/t322/260/1373090020/189626/610b2da/543888a1Ncd81c157.png\" alt=\"\" /></div></div><div class=\"formwork\"><div class=\"formwork_titlecenter\">用手機遙控電視、空調</div><div class=\"formwork_text\">配備紅外發射器,支援2853款裝置小米手機4 內建紅外遙控功能,支援與紅外接受裝置的傳輸協議,可以通過專門為這一功能定製的小米遙控器App,遙控家裡電視、空調等支援紅外協議的家用電器。目前已支援2853款裝置,更多裝置不斷更</div></div><div class=\"formwork\"><div class=\"formwork_img\"><img src=\"http://img20.360buyimg.com/vc/jfs/t343/265/1372124071/66786/5ee90d6e/543888d1N3d0c4330.png\" alt=\"\" /></div></div><div class=\"formwork\"><div class=\"formwork_titlecenter\">MIUI V5</div><div class=\"formwork_text\">全球24種語言版本,6500萬用戶好評如潮MIUI 針對原生 Android 深入系統底層優化,更流暢也更加省電。其次是好看,上千款原創主題,上萬種搭配,讓手機介面千變萬化。更重要的是,它讓你的生活更加方便。自動識別陌生電話,識別騷擾、詐騙電話。它還可以迅速找到你需要的生活服務,就連預約餐館和醫院掛號也能在系統中完成。</div></div><div class=\"formwork\"><div class=\"formwork_img\"><img src=\"http://img20.360buyimg.com/vc/jfs/t298/277/1374037756/194959/d1e9ee70/543888f1Na55c4785.png\" alt=\"\" /></div></div></div>"}
{"id":1231490,"title":"小米4 白色 聯通3G手機","sellPoint":"賣完下櫃!不鏽鋼金屬邊框,5英寸屏超窄邊,驍龍四核2.5GHz處理器,3G RAM,1300W+800W攝像頭!","price":199900,"num":99999,"barcode":null,"image":"https://g-search3.alicdn.com/img/bao/uploaded/i4/i3/TB1fyILc_mWBKNjSZFBXXXxUFXa.jpg_180x180.jpg_.webp","cid":560,"status":1,"created":1425821627000,"updated":1425821627000}