基於redis的setnx()、get()、getset()方法 分布式鎖

來源:互聯網
上載者:User

一.redis命令講解:
setnx()命令:
setnx的含義就是SET if Not Exists,其主要有兩個參數 setnx(key, value)。

該方法是原子的,如果key不存在,則設定當前key成功,返回1;如果當前key已經存在,則設定當前key失敗,返回0。

get()命令:
get(key) 擷取key的值,如果存在,則返回;如果不存在,則返回nil;
getset()命令:
這個命令主要有兩個參數 getset(key, newValue)。該方法是原子的,對key設定newValue這個值,並且返回key原來的舊值。
假設key原來是不存在的,那麼多次執行這個命令,會出現下邊的效果:
1. getset(key, “value1”) 返回nil 此時key的值會被設定為value1
2. getset(key, “value2”) 返回value1 此時key的值會被設定為value2
3. 依次類推。
二.具體的使用步驟如下:
1. setnx(lockkey, 目前時間+到期逾時時間) ,如果返回1,則擷取鎖成功;如果返回0則沒有擷取到鎖,轉向2。
2. get(lockkey)擷取值oldExpireTime ,並將這個value值與當前的系統時間進行比較,如果小於當前系統時間,則認為這個鎖已經逾時,可以允許別的請求重新擷取,轉向3。
3. 計算newExpireTime=目前時間+到期逾時時間,然後getset(lockkey, newExpireTime) 會返回當前lockkey的值currentExpireTime。
4. 判斷currentExpireTime與oldExpireTime 是否相等,如果相等,說明當前getset設定成功,擷取到了鎖。如果不相等,說明這個鎖又被別的請求擷取走了,那麼當前請求可以直接返回失敗,或者繼續重試。
5. 在擷取到鎖之後,當前線程可以開始自己的業務處理,當處理完畢後,比較自己的處理時間和對於鎖設定的逾時時間,如果小於鎖設定的逾時時間,則直接執行delete釋放鎖;如果大於鎖設定的逾時時間,則不需要再鎖進行處理。

具體代碼如下:

實現分布式鎖DistributedLockHandler類:

[java] view plain copy
package tk.mybatis.springboot.distributedLock;

import org.springframework.stereotype.Service;

import redis.clients.jedis.Jedis;

@Service(“distributedLockHandler”)
public class DistributedLockHandler {

private static final Integer Lock_Timeout = 3;  private Jedis jedis;  /**  * 外部調用加鎖的方法  * @param lockKey 鎖的名字  * @param timeout 逾時時間(放置時間長度,如:5L)  * @return  */  public boolean tryLock(String lockKey, Long timeout) {      try {          Long currentTime = System.currentTimeMillis();//開始加鎖的時間          boolean result = false;          while (true) {              if ((System.currentTimeMillis() - currentTime) / 1000 > timeout) {//目前時間超過了設定的逾時時間                  System.out.println("Execute DistributedLockHandler.tryLock method, Time out.");                  break;              } else {                  result = innerTryLock(lockKey);                  if (result) {                      break;                  } else {                      System.out.println("Try to get the Lock,and wait 100 millisecond....");                      Thread.sleep(100);                  }              }          }          return result;      } catch (Exception e) {          System.out.println("Failed to run DistributedLockHandler.getLock method."+ e);          return false;      }  }  /**  * 釋放鎖  * @param lockKey 鎖的名字  */  public void realseLock(String lockKey) {      if(!checkIfLockTimeout(System.currentTimeMillis(), lockKey)){          jedis.del(lockKey);      }  }  /**  * 內部擷取鎖的實現方法  * @param lockKey 鎖的名字  * @return  */  private boolean innerTryLock(String lockKey) {      long currentTime = System.currentTimeMillis();//目前時間      String lockTimeDuration = String.valueOf(currentTime + Lock_Timeout + 1);//鎖的期間      Long result = jedis.setnx(lockKey, lockTimeDuration);      if (result == 1) {          return true;      } else {          if (checkIfLockTimeout(currentTime, lockKey)) {              String preLockTimeDuration = jedis.getSet(lockKey, lockTimeDuration);              if (currentTime > Long.valueOf(preLockTimeDuration)) {                  return true;              }          }          return false;      }  }  /**  * 判斷加鎖是否逾時  * @param currentTime 目前時間  * @param lockKey 鎖的名字  * @return  */  private boolean checkIfLockTimeout(Long currentTime, String lockKey) {      if (currentTime > Long.valueOf(jedis.get(lockKey))) {//目前時間超過鎖的期間          return true;      } else {          return false;      }  }  public DistributedLockHandler setJedis(Jedis jedis) {      this.jedis = jedis;      return this;  }  

}

調用Demo類:

[java] view plain copy
package tk.mybatis.springboot.distributedLock;

import redis.clients.jedis.Jedis;

/**
* 基於redis的setnx()、get()、getset()方法 分布式鎖
* @author KF01
*
*/
public class Demo {
private static final String lockKey = “Lock.TecentIm_Interface_Counter”;

public static void main(String[] args) {      Jedis jedis = new Jedis("127.0.0.1", 6379);      DistributedLockHandler distributedLockHandler = new DistributedLockHandler().setJedis(jedis);      try{          boolean getLock = distributedLockHandler.tryLock(lockKey, Long.valueOf(5));          if(getLock){              // Do your job              System.out.println("Do your job........");          }      }catch(Exception e){          System.out.println(e);      }finally {          distributedLockHandler.realseLock(lockKey);      }  }  

}

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.