memcached 学习笔记 4
memcached真实项目中的应用
1 缓存式的Web应用程序架构
有了缓存的支持,我们可以在传统的app层和db层之间加入cache层,每个app服务器都可以绑定一个mc,
每次数据的读取都可以从ms中取得,如果没有,再从db层读取。
而当数据要进行更新时,除了要发送update的sql给db层,同时也要将更新的数据发给mc,让mc去更新ms中的数据。

假设今后我们的数据库可以和ms进行通讯了,那可以将更新的任务统一交给db层,
每次数据库更新数据的同时会自动去更新ms中的数据,这样就可以进一步减少app层的逻辑复杂度。如下图:

不过每次我们如果没有从cache读到数据,都不得不麻烦数据库。为了最小化数据库的负载压力,我们可以部署数据库复写,
用slave数据库来完成读取操作,而master数据库永远只负责三件事:1.更新数据;2.同步slave数据库;3.更新cache。如下图:

以上这些缓存式web架构在实际应用中被证明是能有效并能极大地降低数据库的负载同时又能提高web的运行性能。
当然这些架构还可以根据具体的应用环境进行变种,以达到不同硬件条件下性能的最优化。
2 以代码为示例
这一篇将以介绍一个memcached在项目中的应用。
假设我们有一个web应用,里面有商品信息,文章信息,评论信息,其他信息,我们希望对其做缓存,
那么我们在ServiceImpl层就不在调用DAOmpl层,而是调用CacheImpl层,
在CacheImpl层中判断要取出的商品信息是否已经在缓存中,如果在了,那么直接从缓存中去,
如果没有这个时候还是从数据库中取,同时将它放到缓存中,以便下次使用。
2.1 新建一个常量类,用于上面的四种信息的在数组中的索引。
public class MemcachedConstant {
public static final int MEMCACHED_GOODSDETAIL = ;
public static final int MEMCACHED_ARTICLEDETAIL = ;
public static final int MEMCACHED_COMMENTDETAIL = ;
public static final int MEMCACHED_OTHERDETAIL = ;
}
2.2 加前缀
由于有大量的商品信息,我们在放入缓存时必须给定一个key,那么我们最好规范的命名不同类别的key,如商品的key就是商品的前缀加上商品的编号。
.public class MemcachedKeyUtil {
. private static final String GOODS_KEY_PREFIX = "goods_";
.
. public static String getGoodsKey(long goodsId) {
. return GOODS_KEY_PREFIX + goodsId;
. }
.}
.3 建立缓存工具类
我们建一个和上一篇文章中一样的工具类,用于新建pool、client,操作缓存等。
这里再强调一下,一个pool关联多个server(就是会根据权重将缓存放在这些servers上),一个client会通过poolName关联具体的pool。
.public class MemcachedUtil {
. private int MEMCACHED_SERVER_NUM = ;
. private SockIOPool[] pools = new SockIOPool[MEMCACHED_SERVER_NUM];
. private MemCachedClient[] mcs = new MemCachedClient[MEMCACHED_SERVER_NUM];
. private final String[] poolNames = new String[] { "GOODSDETAIL_POOL", "", "", "" };
. private static MemcachedUtil instance;
. private MemcachedUtil() {
. this.init();
. }
. // 单例
. public static MemcachedUtil getInstance() {
. if (MemcachedUtil.instance == null) {
. synchronized (MemcachedUtil.class) {
. if (MemcachedUtil.instance == null) {
. MemcachedUtil.instance = new MemcachedUtil();
. }
. }
. }
. return MemcachedUtil.instance;
. }
.
. public Object get(int index, String key) {
. return this.mcs[index].get(key);
. }
.
. public boolean set(int index, String key, Object value) {
. return this.mcs[index].set(key, value);
. }
.
. public boolean delete(String key) {
. return this.mcs[index].delete(key);
. }
. public MemCachedClient getMemCachedClient(int index) {
. return this.mcs[index];
. }
.
. public void init() {
. for (int i = ; i < MEMCACHED_SERVER_NUM; ++i) {
. this.pools[i] = SockIOPool.getInstance(poolNames[i]);
. this.pools[i].setServers(servers);
. this.pools[i].setWeights(weights);
. this.pools[i].setInitConn(initConn);
. this.pools[i].setMinConn(minConn);
. this.pools[i].setMaxConn(maxConn);
. this.pools[i].setMaxIdle(maxIdle);
. this.pools[i].setMaxBusyTime(maxBusyTime);
. this.pools[i].setMaintSleep(maintSleep);
. this.pools[i].setNagle(ifNagle);
. this.pools[i].setSocketTO(socketTO);
. this.pools[i].setSocketConnectTO(socketConnectTO);
. this.pools[i].setFailover(ifFailOver);
. this.pools[i].setFailback(ifFailback);
. this.pools[i].setAliveCheck(ifAliveCheck);
. this.pools[i].initialize();
. this.mcs[i] = new MemCachedClient(poolNames[i]);
. }
. }
.}
2.3 与dao一起使用
新建一个基类以供所用继承它的CacheImpl直接调用MemcachedUtil里的方法,如果不写该类那么在CacheImpl中会有很多重复的操作MemcachedUtil的代码。
.public class MemcachedSupport {
. public boolean setDetailData(String key, Object value) {
. return MemcachedUtil.getInstance().set(MemcachedConstant.MEMCACHED_DETAIL, key, value);
. }
.
. public Object getDetailData(String key) {
. return MemcachedUtil.getInstance().get(MemcachedConstant.MEMCACHED_DETAIL, key);
. }
.
. public boolean deleteDetailData(String key) {
. return MemcachedUtil.getInstance().delete(MemcachedConstant.MEMCACHED_DETAIL);
. }
.}
新建一个GoodsCacheImpl,该类的作用就是一开始所说的,取不到缓存,就调用DAO查询并放入缓存,如果缓存中有就直接从缓存中拿。
1.public class GoodsCacheImpl extends MemcachedSupport{
2. @Resource(name = "goodsDaoImpl")
3. private GoodsDao goodsDao;
4.
5. public Goods selectGoodsById(long goodsId) {
6. Goods goods = null;
7. String goodsKey = MemcachedKeyUtil.getGoodsKey(goodsId);
8. goods = (Goods) getDetailData(goodsKey);
9. if (goods == null) {
10. goods = goodsDao.selectGoodsById(goodsId, false);
11. if (goods != null) {
12. setDetailData(goodsKey, goods);
13. }
14. }
15. return goods;
16. }
17.}
memcached 学习笔记 4的更多相关文章
- memcached学习笔记——存储命令源码分析下篇
上一篇回顾:<memcached学习笔记——存储命令源码分析上篇>通过分析memcached的存储命令源码的过程,了解了memcached如何解析文本命令和mencached的内存管理机制 ...
- memcached学习笔记——存储命令源码分析上篇
原创文章,转载请标明,谢谢. 上一篇分析过memcached的连接模型,了解memcached是如何高效处理客户端连接,这一篇分析memcached源码中的process_update_command ...
- Memcached 学习笔记(二)——ruby调用
Memcached 学习笔记(二)——ruby调用 上一节我们讲述了怎样安装memcached及memcached常用命令.这一节我们将通过ruby来调用memcached相关操作. 第一步,安装ru ...
- Memcached学习笔记
[TOC] 前言 此为学习笔记汇总,如有纰漏之处,还望不吝指出,谢谢. 启动流程 调用settings_init()设定初始化参数 从启动命令中读取参数来设置setting值 设定LIMIT参数 开始 ...
- memcached学习笔记2--安装及命令
学习memcached的原理: 用户一 -> 访问浏览器 -> 服务器Apache -> PHP文件(该文件应用了memcached技术) -> [第一次]到数据库DB中查找数 ...
- memcached学习笔记6--浅谈memcached的机制 以及 memcached细节讨论
附:请浅谈memcached的机制 答: ①基于C/S架构,协议比较简单 c/s架构,此时memcached为服务器端,我们可以使用如PHP,c++/c等程序连接memcached服务器. memca ...
- memcached学习笔记5--socke操作memcached 缓存系统
使用条件:当我们没有权限或者不能使用服务器的时候,我们需要用socket操作memcached memcached-client操作 特点: 无需开启memcache扩展 使用fsocketopen( ...
- memcached学习笔记4--memcache扩展操作memcached
1. 安装并配置memcache扩展库 找到php.ini文件 添加: extendsion= php_memcache.dll 并把对应的dll文件拷贝到ext目录 2. 使用PHP对Memcahc ...
- memcached学习笔记3--telnet操作memcached
方式: 一.telnet访问memcached缓存系统(主要用于教学,不讨论) telnet 127.0.0.1 11211 => telnet IP地址 端口号 //往Memcache ...
- memcached学习笔记1--概念
1.memcached是danga的一个项目,最早是LiveJournal服务的,最初为了加速LiveJournal访问速度而开发,后来被很多大型网站采用 官网: http://www.danga.c ...
随机推荐
- tomcat-java_opts设置说明
The JAVA_OPTS environment variable can be used to specify additional arguments to the JVM JBoss will ...
- struts2拦截器demo
按照网上的一些资料配置的,期间也出现过几个错误. 其中有个错误,是关于struts.xml里面package配置的问题,因为里面的几个标签是有顺序的. 顺序是: result-types interc ...
- 使用python登录CNZZ访问量统计网站,然后获取相应的数据
思路: 第一步:使用pypeteer.launcher打开浏览器, 第二步:向CNZZ的登录(通过使用iframe嵌入的阿里巴巴单点登录页面),向iframe页面中自动输入用户名和密码,然后点击登录按 ...
- WPF绑定BitMapImage
先说下图片文件存在服务器.wpf常用绑定图片地址没办法用.忽然想到,convert能否转字节数据?实验了下可以. 图片绑定字节数组. convert代码 public class PictureCon ...
- LightOJ 1197 LightOJ 1197(大区间素数筛选)
http://lightoj.com/volume_showproblem.php?problem=1197 题目大意: 就是给你一个区间[a,b]让你求这个区间素数的个数 但a.b的值太大没法直接进 ...
- Windows7 64位下SDK Manager.exe无法运行问题解决方法
我在Windows7 64位下运行SDK Manager.exe总是一闪而过,无法正常启动它,最后在网上找到一篇文章,修改系统“path”变量,把“path”变量中的第一项设置为我的JDK目录“C:\ ...
- 红与黑(DFS)
描述有一间长方形的房子,地上铺了红色.黑色两种颜色的正方形瓷砖.你站在其中一块黑色的瓷砖上,只能向相邻的黑色瓷砖移动.请写一个程序,计算你总共能够到达多少块黑色的瓷砖.输入包括多个数据集合.每个数据集 ...
- 关于JNDI技术链接操作数据库-2019.1.10
Java 命名与目录接口(Java Naming and Directory Interface) ==需要使用的包为java.sql&javax.naming包==Mysql 步骤: 1.配 ...
- java异常术语
.检查性异常: 不处理编译不能通过 .非检查性异常:不处理编译可以通过,如果有抛出直接抛到控制台(所有RuntimeException的派生类都是非检查型异常) .非运行时异常: 就是检查性异常4.运 ...
- robot framework学习笔记之一 资源文件(Resource)和外部资源(External Resources)
一.资源文件(Resource) 测试套件主要是存放测试案例,资源文件主要是用来存放用户关键字. 添加资源 在目录型的Project/Test Suite下单击鼠标右键,选择『New Resou ...