Golang示例续期锁:Redis+Channel+sync.Mutex

package main

import (
	"context"
	"fmt"
	"github.com/go-redis/redis"
	"golang.org/x/sync/errgroup"
	"log"
	"sync"
	"time"
)

func main() {
	NewRedis()
	fu1()

	time.Sleep(100 * time.Second)
}

var rdb *redis.Client
var ctx = context.Background()
var mutex sync.Mutex

func NewRedis() {
	rdb = redis.NewClient(&redis.Options{
		Addr:     "localhost:6379",
		Password: "", // no password set
		DB:       0,  // use default DB
	})
}
func Lock(key string) error {
	mutex.Lock()
	defer mutex.Unlock()
	_, err := rdb.SetNX(key, 1, 1*time.Second).Result()
	if err != nil {
		log.Println(err.Error())
	}
	return err
}
func UnLock(key string) error {
	_, err := rdb.Del(key).Result()
	if err != nil {
		log.Println(err.Error())
		return err
	}
	return err
}
func Expire(key string) error {
	_, err := rdb.Expire(key, 1*time.Second).Result()
	if err != nil {
		log.Println(err.Error())
		return err
	}
	return err
}
func fu1() error {
	ch := make(chan bool)
	// 加锁
	err := Lock("lock_key")
	if err != nil {
		return err
	}
	//解锁
	defer func() {
		err = UnLock("lock_key")
		if err != nil {
			fmt.Println(err.Error())
			return
		}
		fmt.Println("release redis lock success")
	}()

	g, _ := errgroup.WithContext(context.Background())
	//...主业务代码
	g.Go(func() error {
		time.Sleep(15 * time.Second)
		ch <- true
		return nil
	})

	//锁续期
	g.Go(func() error {
		ticker := time.NewTicker(time.Second * 1)
		for {
			select {
			// 任务还没执行完 每秒续期
			case <-ticker.C:
				Expire("lock_key")
				fmt.Println(time.Now())

				//收到执行完的 channel 就关闭time定时任务
			case <-ch:
				ticker.Stop()
				return nil
			}

		}
	})
	//等待信号量
	if err = g.Wait(); err != nil {
		return err
	}
	close(ch)
	return nil
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值