前言

Memcached 是一个高性能的分布式内存对象缓存系统,其存储性能在某些方面不比redis差,甚至在文本类型数据的存储上性能略优于redis,本文将介绍如何在springboot中集成memcached

准备工作

首先我们需要一款客户端连接对象,类似于我们在使用redis时使用的jedis , 目前memcached主流的客户端是Xmemcached,如果使用的是maven构建项目,则引入对应的依赖

   <!--引入memcached-->
<dependency>
<groupId>com.googlecode.xmemcached</groupId>
<artifactId>xmemcached</artifactId>
<version>2.4.5</version>
</dependency>

配置

memcached在yml(properties)文件的相关配置

# memcached配置
memcached:
server: 1.1.1.1:3333 2.2.2.2:4444 #memcached服务器集群(格式为host:port,多个服务器之间用空格隔开)
opTimeout: 3000 #接口操作的默认超时时间,可以被接口覆盖
poolSize: 10 #池子大小
failureMode: false #是否开启失败模式,默认为false
enabled: true # 是否使用memcached缓存

将memcached的配置由bean来管理,方便我们调用

 package com.me.config.properties;

 import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component; /**
* @author : wang zns
* @date : 2018-12-19
*/
@Component
@ConfigurationProperties(prefix = "memcached")
public class MemcachedProperties { /**
* 服务器
*/
private String server; /**
* 操作超时时间,可以被API覆盖
*/
private Integer opTimeout;
/**
* 连接池大小
*/
private Integer poolSize; /**
* 是否开启失败模式
*/
private boolean failureMode; /**
* 是否使用memcached缓存
*/
private boolean enabled; public String getServer() {
return server;
} public void setServer(String server) {
this.server = server;
} public Integer getOpTimeout() {
return opTimeout;
} public void setOpTimeout(Integer opTimeout) {
this.opTimeout = opTimeout;
} public Integer getPoolSize() {
return poolSize;
} public void setPoolSize(Integer poolSize) {
this.poolSize = poolSize;
} public boolean isFailureMode() {
return failureMode;
} public void setFailureMode(boolean failureMode) {
this.failureMode = failureMode;
} public boolean isEnabled() {
return enabled;
} public void setEnabled(boolean enabled) {
this.enabled = enabled;
}
}

memcached配置类(创建memcached客户端对象,并注入spring容器中)

 package com.me.config;

 import cn.stylefeng.guns.config.properties.MemcachedProperties;
import lombok.extern.slf4j.Slf4j;
import net.rubyeye.xmemcached.MemcachedClient;
import net.rubyeye.xmemcached.MemcachedClientBuilder;
import net.rubyeye.xmemcached.XMemcachedClientBuilder;
import net.rubyeye.xmemcached.command.BinaryCommandFactory;
import net.rubyeye.xmemcached.impl.KetamaMemcachedSessionLocator;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration; /**
* @author : wang zns
* @date : 2018-12-19
*/
@Configuration
@Slf4j
public class MemcachedConfig { @Autowired
private MemcachedProperties memcachedProperties; @Bean(name = "memcachedClientBuilder")
public MemcachedClientBuilder getBuilder() {
MemcachedClientBuilder memcachedClientBuilder = new XMemcachedClientBuilder(memcachedProperties.getServer()); // 内部采用一致性哈希算法
memcachedClientBuilder.setSessionLocator(new KetamaMemcachedSessionLocator());
// 操作的超时时间
memcachedClientBuilder.setOpTimeout(memcachedProperties.getOpTimeout());
// 采用二进制传输协议(默认为文本协议)
memcachedClientBuilder.setCommandFactory(new BinaryCommandFactory());
// 设置连接池的大小
memcachedClientBuilder.setConnectionPoolSize(memcachedProperties.getPoolSize());
// 是否开起失败模式
memcachedClientBuilder.setFailureMode(memcachedProperties.isFailureMode());
return memcachedClientBuilder;
} /**
* 由Builder创建memcachedClient对象,并注入spring容器中
* @param memcachedClientBuilder
* @return
*/
@Bean(name = "memcachedClient")
public MemcachedClient getClient(@Qualifier("memcachedClientBuilder") MemcachedClientBuilder memcachedClientBuilder) {
MemcachedClient client = null;
try {
client = memcachedClientBuilder.build();
} catch(Exception e) {
log.info("exception happens when bulid memcached client{}",e.toString());
}
return client;
} }

使用

有了client对象之后,我们可以像直接操作服务端那样进行对应的操作,下面用一个小案例进行演示

service

 package cn.stylefeng.guns.modular.housemanage.cache;

 import cn.stylefeng.guns.modular.system.model.TblHouse;

 /**
* 房屋管理缓存 业务层
* @author : wang zns
* @date : 2018-12-19
*/
public interface HouseManageCacheService { /**
* 添加
* @param tblHouse
*/
void add(TblHouse tblHouse); /**
* 根据主键删除
* @param tblHouseId
*/
void delete(Integer tblHouseId); }

serviceImpl

 package cn.stylefeng.guns.modular.housemanage.cache;

 import cn.stylefeng.guns.modular.housemanage.service.ITblHouseService;
import cn.stylefeng.guns.modular.system.model.TblHouse;
import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import net.rubyeye.xmemcached.MemcachedClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service; /**
* @author : wang zns
* @date : 2018-12-19
*/
@Service
@Slf4j
public class HouseManageCacheServiceImpl implements HouseManageCacheService{ @Autowired
private MemcachedClient memcachedClient;
@Autowired
private ITblHouseService iTblHouseService; @Override
public void add(TblHouse tblHouse) {
// 先入库,入库成功则入缓存
boolean isSuccess = iTblHouseService.insert(tblHouse);
if (isSuccess) {
try {
String houseJsonStr = JSON.toJSONString(tblHouse);
memcachedClient.set(String.valueOf(tblHouse.getId()),0,houseJsonStr);
} catch (Exception e) {
log.info("exception happens when add House:{}",e.toString());
throw new RuntimeException(e.getMessage());
}
}
} @Override
public void delete(Integer tblHouseId) {
// 先删除数据库内容,成功则清空缓存
boolean isSuccess = iTblHouseService.deleteById(tblHouseId);
if (isSuccess) {
try {
memcachedClient.delete(String.valueOf(tblHouseId));
} catch (Exception e) {
log.info("exception happens when delete House:{}",e.toString());
throw new RuntimeException(e.getMessage());
}
}
} }

controller

  /**
* 新增房屋管理
*/
@RequestMapping(value = "/add")
@ResponseBody
public Object add(@Valid TblHouse tblHouse, BindingResult bindingResult) {
if(bindingResult.hasErrors()){
throw new ServiceException(BizExceptionEnum.REQUEST_NULL);
}
// 如果确定要使用缓存
if (memcachedProperties.isEnabled()) {
houseManageCacheService.add(tblHouse);
} else {
tblHouseService.insert(tblHouse);
}
return SUCCESS_TIP;
} /**
* 删除房屋管理
*/
@RequestMapping(value = "/delete")
@ResponseBody
public Object delete(@RequestParam Integer tblHouseId) {
if (memcachedProperties.isEnabled()) {
houseManageCacheService.delete(tblHouseId);
} else {
tblHouseService.deleteById(tblHouseId);
}
return SUCCESS_TIP;
}

运行结果:

提交之后,去缓存服务器上查看

至此,springboot中集成memcached就完成了

写在最后

springboot集成memcached非常简单,核心步骤就是添加依赖、创建client对象并注入spring容器、然后就是用client对象进行各种操作。

client的接口当然有非常多,xmemcached对接口进行了封装。

最后附上xmemcached官网的地址,里面的文档很详细  https://github.com/killme2008/xmemcached/wiki

springboot中集成memcached的更多相关文章

  1. Discuz!NT中集成Memcached分布式缓存

    大约在两年前我写过一篇关于Discuz!NT缓存架构的文章,在那篇文章的结尾介绍了在IIS中如果开启多个应用程序池会造成多个缓存实例之间数据同步的问题.虽然给出了一个解决方案,但无形中却把压力转移到了 ...

  2. 在springboot中集成mybatis开发

    在springboot中利用mybatis框架进行开发需要集成mybatis才能进行开发,那么如何在springboot中集成mybatis呢?按照以下几个步骤就可以实现springboot集成myb ...

  3. 在springboot中集成jsp开发

    springboot就是一个升级版的spring.它可以极大的简化xml配置文件,可以采用全注解形式开发,一个字就是很牛.在springboot想要使用jsp开发,需要集成jsp,在springboo ...

  4. SpringBoot中集成redis

    转载:https://www.cnblogs.com/zeng1994/p/03303c805731afc9aa9c60dbbd32a323.html 不是使用注解而是代码调用 需要在springbo ...

  5. SpringBoot中集成Swagger2

    1.依赖jar <dependency> <groupId>io.springfox</groupId> <artifactId>springfox-s ...

  6. 如何在SpringBoot中集成JWT(JSON Web Token)鉴权

    这篇博客主要是简单介绍了一下什么是JWT,以及如何在Spring Boot项目中使用JWT(JSON Web Token). 1.关于JWT 1.1 什么是JWT 老生常谈的开头,我们要用这样一种工具 ...

  7. springboot 中 集成druid ,redis

    1,导入druid jar包 <!--引入drud--> <dependency> <groupId>com.alibaba</groupId> < ...

  8. 使用Logstash同步数据至Elasticsearch,Spring Boot中集成Elasticsearch实现搜索

    安装logstash.同步数据至ElasticSearch 为什么使用logstash来同步,CSDN上有一篇文章简要的分析了以下几种同步工具的优缺点:https://blog.csdn.net/la ...

  9. springboot中使用spring-session实现共享会话到redis(二)

    上篇文章介绍了springboot中集成spring-session实现了将session分布式存到redis中.这篇在深入介绍一些spring-session的细节. 1.session超时: 在t ...

随机推荐

  1. python + excel工资条自动生成

    年终绩效分配结果出来了,领导要求每人要清楚地知道自己的情况.要求:总绩效和各分类都要清楚.这就表示我们要给每人六个纸条,一个总的,五个分的.打出来,裁开,分发给每个人!累死人.所以,我就想能否每人生成 ...

  2. 通过sd文件发布的FeatureAccess服务不能查看到图层

    发布服务有两种方法, 1. 用ArcMap --Share As - service --publish a service 此方法可以直接将地图数据发布到ArcGIS  Server 的地图服务中, ...

  3. SpringCloud(一)

    什么是SpringCloud? Spring Cloud是一系列框架的有序集合.它利用Spring Boot的开发便利性巧妙地简化了分布式系统基础设施的开发,如服务发现注册.配置中心.消息总线.负载均 ...

  4. SpringBoot整合持久层技术--(一)JdbcTemplate

    简介: JdbcTemplate是Spring提供的一套JDBC模板框架,利用AOP技术解决直接使用JDBC带来的重复代码问题.它没有MyBatis使用那么灵活,但是却比直接使用JDBC方便得多.Sp ...

  5. 【Spring】利用spring的JdbcTemplate查询返回结果映射到自定义类型

    // org.springframework.jdbc.core.JdbcTemplate 中的查询方法基本都有支持参数RowMapper<T> rowMapper的重载方法.下面只是随便 ...

  6. 嵊州D6T2 城市 city

    城市 city [问题描述] 众所周知,why 是czyz 王国的国王. czyz 王国一共有n 个城市,每个城市都有一条道路连向一个城市(可能连向这个城市自己). 同时,对于每一个城市,也只有一条道 ...

  7. Typecho的卡哇伊小猫咪小插件(Live2D猫咪插件)

    之前看到一个博客,被它博客上的动态小猫咪给吸引了,这个纯粹就是一个在线撸猫的神器啊.但是在网上寻找一番,并没有找到合适的插件,或者说没有找到合适的模型,因此无奈之后,只能向该博主请教,在它写的博客上, ...

  8. JS 百度地图-右键菜单

    JS 百度地图-右键菜单 /*-----------------标注右键删除-------------------------*/ var markerMenu = new BMap.ContextM ...

  9. MyBatis 中 Mapper 接口的使用原理

    MyBatis 中 Mapper 接口的使用原理 MyBatis 3 推荐使用 Mapper 接口的方式来执行 xml 配置中的 SQL,用起来很方便,也很灵活.在方便之余,想了解一下这是如何实现的, ...

  10. Wormholes POJ - 3259 spfa判断负环

    //判断负环 dist初始化为正无穷 //正环 负无穷 #include<iostream> #include<cstring> #include<queue> # ...