SpringBoot基於數據庫實現簡單的分佈式鎖

本文介紹SpringBoot基於數據庫實現簡單的分佈式鎖。java

1.簡介

分佈式鎖的方式有不少種,一般方案有:mysql

  • 基於mysql數據庫
  • 基於redis
  • 基於ZooKeeper

網上的實現方式有不少,本文主要介紹的是若是使用mysql實現簡單的分佈式鎖,加鎖流程以下圖:git

其實大體思想以下:web

  • 1.根據一個值來獲取鎖(也就是我這裏的tag),若是當前不存在鎖,那麼在數據庫插入一條記錄,而後進行處理業務,當結束,釋放鎖(刪除鎖)。
  • 2.若是存在鎖,判斷鎖是否過時,若是過時則更新鎖的有效期,而後繼續處理業務,當結束時,釋放鎖。若是沒有過時,那麼獲取鎖失敗,退出。

2.數據庫設計

2.1 數據表介紹

數據庫表是由JPA自動生成的,稍後會對實體進行介紹,內容以下:redis

CREATE TABLE `lock_info` (
  `id` bigint(20) NOT NULL,
  `expiration_time` datetime NOT NULL,
  `status` int(11) NOT NULL,
  `tag` varchar(255) NOT NULL,
  PRIMARY KEY (`id`),
  UNIQUE KEY `uk_tag` (`tag`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci;
複製代碼

其中:spring

  • id:主鍵
  • tag:鎖的標示,以訂單爲例,能夠鎖訂單id
  • expiration_time:過時時間
  • status:鎖狀態,0,未鎖,1,已經上鎖

3.實現

本文使用SpringBoot 2.0.3.RELEASE,MySQL 8.0.16,ORM層使用的JPA。sql

3.1 pom

新建項目,在項目中加入jpa和mysql依賴,完整內容以下:數據庫

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>2.0.3.RELEASE</version>
		<relativePath/> <!-- lookup parent from repository -->
	</parent>
	<groupId>com.dalaoyang</groupId>
	<artifactId>springboot2_distributed_lock_mysql</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<name>springboot2_distributed_lock_mysql</name>
	<description>springboot2_distributed_lock_mysql</description>

	<properties>
		<java.version>1.8</java.version>
	</properties>

	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-web</artifactId>
		</dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-data-jpa</artifactId>
		</dependency>

		<dependency>
			<groupId>mysql</groupId>
			<artifactId>mysql-connector-java</artifactId>
			<scope>runtime</scope>
		</dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>

		<dependency>
			<groupId>org.projectlombok</groupId>
			<artifactId>lombok</artifactId>
			<version>1.16.22</version>
			<scope>provided</scope>
		</dependency>
	</dependencies>

	<build>
		<plugins>
			<plugin>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-maven-plugin</artifactId>
			</plugin>
		</plugins>
	</build>

</project>
複製代碼

3.2 配置文件

配置文件配置了一下數據庫信息和jpa的基本配置,以下:apache

server.port=20001


##數據庫配置
##數據庫地址
spring.datasource.url=jdbc:mysql://localhost:3306/lock?characterEncoding=utf8&useSSL=false
##數據庫用戶名
spring.datasource.username=root
##數據庫密碼
spring.datasource.password=12345678
##數據庫驅動
spring.datasource.driver-class-name=com.mysql.jdbc.Driver


##validate 加載hibernate時,驗證建立數據庫表結構
##create 每次加載hibernate,從新建立數據庫表結構,這就是致使數據庫表數據丟失的緣由。
##create-drop 加載hibernate時建立,退出是刪除表結構
##update 加載hibernate自動更新數據庫結構
##validate 啓動時驗證表的結構,不會建立表
##none 啓動時不作任何操做
spring.jpa.hibernate.ddl-auto=update

##控制檯打印sql
spring.jpa.show-sql=true
##設置innodb
spring.jpa.database-platform=org.hibernate.dialect.MySQL5InnoDBDialect

複製代碼

3.3 實體類

實體類以下,這裏給tag字段設置了惟一索引,防止重複插入相同的數據:springboot

package com.dalaoyang.entity;


import lombok.Data;
import javax.persistence.*;
import java.util.Date;

@Data
@Entity
@Table(name = "LockInfo",
        uniqueConstraints={@UniqueConstraint(columnNames={"tag"},name = "uk_tag")})
public class Lock {

    public final static Integer LOCKED_STATUS = 1;
    public final static Integer UNLOCKED_STATUS = 0;

    /**
     * 主鍵id
     */
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;

    /**
     * 鎖的標示,以訂單爲例,能夠鎖訂單id
     */
    @Column(nullable = false)
    private String tag;

    /**
     * 過時時間
     */
    @Column(nullable = false)
    private Date expirationTime;

    /**
     * 鎖狀態,0,未鎖,1,已經上鎖
     */
    @Column(nullable = false)
    private Integer status;

    public Lock(String tag, Date expirationTime, Integer status) {
        this.tag = tag;
        this.expirationTime = expirationTime;
        this.status = status;
    }

    public Lock() {
    }
}
複製代碼

3.4 repository

repository層只添加了兩個簡單的方法,根據tag查找鎖和根據tag刪除鎖的操做,內容以下:

package com.dalaoyang.repository;

import com.dalaoyang.entity.Lock;
import org.springframework.data.jpa.repository.JpaRepository;


public interface LockRepository extends JpaRepository<Lock, Long> {

    Lock findByTag(String tag);

    void deleteByTag(String tag);
}

複製代碼

3.5 service

service接口定義了兩個方法,獲取鎖和釋放鎖,內容以下:

package com.dalaoyang.service;


public interface LockService {

    /**
     * 嘗試獲取鎖
     * @param tag 鎖的鍵
     * @param expiredSeconds 鎖的過時時間(單位:秒),默認10s
     * @return
     */
    boolean tryLock(String tag, Integer expiredSeconds);

    /**
     * 釋放鎖
     * @param tag 鎖的鍵
     */
    void unlock(String tag);
}

複製代碼

實現類對上面方法進行了實現,其內容與上述流程圖中一致,這裏不在作介紹,完整內容以下:

package com.dalaoyang.service.impl;

import com.dalaoyang.entity.Lock;
import com.dalaoyang.repository.LockRepository;
import com.dalaoyang.service.LockService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.StringUtils;

import java.util.Calendar;
import java.util.Date;
import java.util.Objects;


@Service
public class LockServiceImpl implements LockService {

    private final Integer DEFAULT_EXPIRED_SECONDS = 10;

    @Autowired
    private LockRepository lockRepository;

    @Override
    @Transactional(rollbackFor = Throwable.class)
    public boolean tryLock(String tag, Integer expiredSeconds) {
        if (StringUtils.isEmpty(tag)) {
            throw new NullPointerException();
        }
        Lock lock = lockRepository.findByTag(tag);
        if (Objects.isNull(lock)) {
            lockRepository.save(new Lock(tag, this.addSeconds(new Date(), expiredSeconds), Lock.LOCKED_STATUS));
            return true;
        } else {
            Date expiredTime = lock.getExpirationTime();
            Date now = new Date();
            if (expiredTime.before(now)) {
                lock.setExpirationTime(this.addSeconds(now, expiredSeconds));
                lockRepository.save(lock);
                return true;
            }
        }
        return false;
    }

    @Override
    @Transactional(rollbackFor = Throwable.class)
    public void unlock(String tag) {
        if (StringUtils.isEmpty(tag)) {
            throw new NullPointerException();
        }
        lockRepository.deleteByTag(tag);
    }

    private Date addSeconds(Date date, Integer seconds) {
        if (Objects.isNull(seconds)){
            seconds = DEFAULT_EXPIRED_SECONDS;
        }
        Calendar calendar = Calendar.getInstance();
        calendar.setTime(date);
        calendar.add(Calendar.SECOND, seconds);
        return calendar.getTime();
    }
}

複製代碼

3.6 測試類

建立了一個測試的controller進行測試,裏面寫了一個test方法,方法在獲取鎖的時候會sleep 2秒,便於咱們進行測試。完整內容以下:

package com.dalaoyang.controller;

import com.dalaoyang.service.LockService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;


@RestController
public class TestController {

    @Autowired
    private LockService lockService;

    @GetMapping("/tryLock")
    public Boolean tryLock(String tag, Integer expiredSeconds) {
        return lockService.tryLock(tag, expiredSeconds);
    }

    @GetMapping("/unlock")
    public Boolean unlock(String tag) {
        lockService.unlock(tag);
        return true;
    }

    @GetMapping("/test")
    public String test(String tag, Integer expiredSeconds) {
        if (lockService.tryLock(tag, expiredSeconds)) {
            try {
                //do something
                //這裏使用睡眠兩秒,方便觀察獲取不到鎖的狀況
                Thread.sleep(2000);
            } catch (Exception e) {

            } finally {
                lockService.unlock(tag);
            }
            return "獲取鎖成功,tag是:" + tag;
        }
        return "當前tag:" + tag + "已經存在鎖,請稍後重試!";
    }
}

複製代碼

3.測試

項目使用maven打包,分別使用兩個端口啓動,分別是20000和20001。

java -jar springboot2_distributed_lock_mysql-0.0.1-SNAPSHOT.jar --server.port=20001
複製代碼
java -jar springboot2_distributed_lock_mysql-0.0.1-SNAPSHOT.jar --server.port=20000
複製代碼

分別訪問兩個端口的項目,如圖所示,只有一個請求能夠獲取鎖。

4.總結

本案例實現的分佈式鎖只是一個簡單的實現方案,還具有不少問題,不適合生產環境使用。

5.源碼地址

源碼地址:gitee.com/dalaoyang/s…

相關文章
相關標籤/搜索