直接上代码,实现原理,见代码中的注释。
package persistent.prestige.redis.lock;
import persistent.prestige.redis.RedisUtils;
import redis.clients.jedis.Jedis;
/**
* 该类实现一个基本的锁,非公平的抢占式的自旋的分布式锁
* 原理:利用redis 提供的 setnx 命令与 del 命令实现分布式锁
* setnx(key,val) 如果key不存在,则将key放入redis,并返回long类型的1,如果key存在,不会改变key的值,并返回long类型的0
* del(key) 释放锁通过该命令,将key删除
*
* @author dingwei2
*
*/
public class Locker
implements Lock {
private static final String LOCK_RESOURCE_KEY = "persistent:prestige:redis:lock"
;
private String lockKey;
public Locker() {
this.lockKey =
LOCK_RESOURCE_KEY;
}
public Locker(String lockKey) {
this.lockKey =
lockKey;
}
/**
* 锁定逻辑 redis getset命令,设置新值,并返回旧值
*
*
*
*/
@Override
public void lock() {
System.out.println(Thread.currentThread().getName() + "准备获取锁"
);
// TODO Auto-generated method stub
Jedis jedis =
null;
try {
jedis =
RedisUtils.getJedis();
while (jedis.setnx(
this.lockKey, "1").longValue() == 0
) {
System.out.println(Thread.currentThread().getName() + "自旋转"
);
}
} finally {
if(jedis !=
null)
RedisUtils.returnResource(jedis);
}
System.out.println(Thread.currentThread().getName() + "获取锁"
);
}
@Override
public void unlock() {
// TODO Auto-generated method stub
System.out.println(Thread.currentThread().getName() + "准备释放锁"
);
Jedis jedis =
null;
try {
jedis =
RedisUtils.getJedis();
jedis.del(this.lockKey);
} finally {
if(jedis !=
null)
RedisUtils.returnResource(jedis);
}
System.out.println(Thread.currentThread().getName() + "释放锁"
);
}
}
附Lock接口
package persistent.prestige.redis.lock;
import java.io.Serializable;
public interface Lock
extends Serializable {
/**
* 锁定
*/
void lock();
/**
* 获取锁
*/
void unlock();
}
附RedisUtil,非原创
package persistent.prestige.redis;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;
public class RedisUtils {
// Redis服务器IP
//private static String ADDR = "10.88.7.3";
private static String ADDR = "192.168.1.103"
;
// Redis的端口号
private static int PORT = 6379
;
// 访问密码
private static String AUTH = "admin"
;
// 可用连接实例的最大数目,默认值为8;
// 如果赋值为-1,则表示不限制;如果pool已经分配了maxActive个jedis实例,则此时pool的状态为exhausted(耗尽)。
private static int MAX_ACTIVE = 1024
;
// 控制一个pool最多有多少个状态为idle(空闲的)的jedis实例,默认值也是8。
private static int MAX_IDLE = 200
;
// 等待可用连接的最大时间,单位毫秒,默认值为-1,表示永不超时。如果超过等待时间,则直接抛出JedisConnectionException;
private static int MAX_WAIT = 10000
;
private static int TIMEOUT = 10000
;
// 在borrow一个jedis实例时,是否提前进行validate操作;如果为true,则得到的jedis实例均是可用的;
private static boolean TEST_ON_BORROW =
true;
private static JedisPool jedisPool =
null;
/**
* 初始化Redis连接池
*/
static {
try {
JedisPoolConfig config =
new JedisPoolConfig();
config.setMaxIdle(MAX_IDLE);
config.setTestOnBorrow(TEST_ON_BORROW);
config.setMaxIdle(200
);
//最大连接数, 应用自己评估,不要超过AliCloudDB for Redis每个实例最大的连接数
config.setMaxTotal(300
);
config.setTestOnBorrow(false);
config.setTestOnReturn(false);
jedisPool =
new JedisPool(config, ADDR, PORT, TIMEOUT);
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 获取Jedis实例
*
* @return
*/
public synchronized static Jedis getJedis() {
try {
if (jedisPool !=
null) {
Jedis resource =
jedisPool.getResource();
return resource;
} else {
return null;
}
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
public static void returnResource(
final Jedis jedis) {
if (jedis !=
null) {
jedisPool.returnResource(jedis);
}
}
public static void main(String[] args) {
// TODO Auto-generated method stub
Jedis jedis =
null;
try {
jedis =
RedisUtils.getJedis();
String key = "persistent:prestige:redis:lock:ReentrantLock:list"
;
System.out.println(jedis.lindex(key, 0 ).equals("0"
));
} finally {
RedisUtils.returnResource(jedis);
}
}
}
转载于:https://www.cnblogs.com/dingwmz/p/5690621.html
相关资源:redis分布式锁工具类