Redis分布式锁

it2022-05-05  153

直接上代码,实现原理,见代码中的注释。 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分布式锁工具类

最新回复(0)