MemcacheCacheStorager.java

package com.foxinmy.weixin4j.cache;

import java.util.ArrayList;
import java.util.Date;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

import com.whalin.MemCached.MemCachedClient;
import com.whalin.MemCached.SockIOPool;

/**
 * 用Memcache保存缓存对象(推荐使用)
 *
 * @className MemcacheCacheStorager
 * @author jinyu(foxinmy@gmail.com)
 * @date 2016年5月11日
 * @since JDK 1.6
 * @see
 */
public class MemcacheCacheStorager<T extends Cacheable> implements
		CacheStorager<T> {

	private final MemCachedClient mc;

	public MemcacheCacheStorager() {
		this(new MemcachePoolConfig());
	}

	public MemcacheCacheStorager(MemcachePoolConfig poolConfig) {
		mc = new MemCachedClient();
		poolConfig.initSocketIO();
		initializeKey();
	}

	@SuppressWarnings("unchecked")
	private Set<String> initializeKey() {
		Set<String> all = (Set<String>) mc.get(ALLKEY);
		if (all == null) {
			all = new HashSet<String>();
			mc.set(ALLKEY, all);
		}
		return all;
	}

	@SuppressWarnings("unchecked")
	@Override
	public T lookup(String key) {
		return (T) mc.get(key);
	}

	@Override
	public void caching(String key, T cache) {
		if (cache.getCreateTime() > 0l) {
			mc.set(key,
					cache,
					new Date(cache.getCreateTime() + cache.getExpires() - CUTMS));
		} else {
			mc.set(key, cache);
		}
		Set<String> all = initializeKey();
		all.add(key);
		mc.set(ALLKEY, all);
	}

	@Override
	public T evict(String key) {
		T cache = lookup(key);
		mc.delete(key);
		Set<String> all = initializeKey();
		all.remove(key);
		mc.set(ALLKEY, all);
		return cache;
	}

	@Override
	public void clear() {
		Set<String> all = initializeKey();
		for (String key : all) {
			mc.delete(key);
		}
		mc.delete(ALLKEY);
	}

	public static class MemcachePoolConfig {
		public final static String HOST = "127.0.0.1";
		public final static int PORT = 11211;
		public final static int WEIGHT = 1;

		public static int minConn = 5;
		public static int initConn = 5;
		public static int maxConn = 100;
		public static int maxIdle = 300000;
		public static long maxBusyTime = 30000L;
		public static int socketTO = 3000;
		public static int socketConnectTO = 3000;
		public static boolean failover = true;
		public static boolean failback = true;
		public static boolean nagle = false;
		public static boolean aliveCheck = false;
		public static final int consistentHash = 3;
		public static final int mainSleep = 30000;

		private static SockIOPool pool;
		static {
			pool = SockIOPool.getInstance();
			pool.setFailback(failback);
			pool.setFailover(failover);
			pool.setMaxBusyTime(maxBusyTime);
			pool.setMaxConn(maxConn);
			pool.setMaxIdle(maxIdle);
			pool.setMinConn(minConn);
			pool.setNagle(nagle);
			pool.setSocketConnectTO(socketConnectTO);
			pool.setSocketTO(socketTO);
			pool.setAliveCheck(aliveCheck);
			pool.setHashingAlg(consistentHash);
			pool.setInitConn(initConn);
			pool.setMaintSleep(maxBusyTime);
		}

		private List<MemcacheConfig> configs;
		private String[] servers;
		private Integer[] weights;

		/**
		 * {localhost:11211,1}
		 */
		public MemcachePoolConfig() {
			this(HOST, PORT, WEIGHT);
		}

		/**
		 * {host:11211,1}
		 *
		 * @param host
		 *            主机
		 */
		public MemcachePoolConfig(String host) {
			this(host, PORT);
		}

		/**
		 * {host:port,1}
		 *
		 * @param host
		 *            主机
		 * @param port
		 *            端口
		 */
		public MemcachePoolConfig(String host, int port) {
			this(host, port, WEIGHT);
		}

		/**
		 * {host:port,weight}
		 *
		 * @param host
		 *            主机
		 * @param port
		 *            端口
		 * @param weight
		 *            权重
		 */
		public MemcachePoolConfig(String host, int port, int weight) {
			configs = new ArrayList<MemcacheConfig>();
			configs.add(new MemcacheConfig(host, port, weight));
		}

		public MemcachePoolConfig addServer(String host) {
			return addServer(host, PORT, WEIGHT);
		}

		public MemcachePoolConfig addServer(String host, int port) {
			return addServer(host, port, WEIGHT);
		}

		public MemcachePoolConfig addServer(String host, int port, int weight) {
			configs.add(new MemcacheConfig(host, port, weight));
			return this;
		}

		private void initConfig() {
			if (servers == null || weights == null) {
				servers = new String[configs.size()];
				weights = new Integer[configs.size()];
				for (int i = 0; i < configs.size(); i++) {
					servers[i] = configs.get(i).getServer();
					weights[i] = configs.get(i).getWeight();
				}
			}
		}

		private void initSocketIO() {
			pool.setServers(getServers());
			pool.setWeights(getWeights());
			if (pool.isInitialized()) {
				pool.shutDown();
			}
			pool.initialize();
		}

		public String[] getServers() {
			initConfig();
			return servers;
		}

		public Integer[] getWeights() {
			initConfig();
			return weights;
		}

		@Override
		public String toString() {
			return "MemcachePoolConfig [" + configs + "]";
		}

		private static class MemcacheConfig {
			private String host;
			private int port;
			private int weight;

			public MemcacheConfig(String host, int port, int weight) {
				this.host = host;
				this.port = port;
				this.weight = weight;
			}

			public String getServer() {
				return String.format("%s:%d", host, port);
			}

			public int getWeight() {
				return weight;
			}

			@Override
			public String toString() {
				return String.format("{%s:%d,%d}", host, port, weight);
			}
		}
	}
}