Golang的memcache如何简单实现


本文摘自php中文网,作者藏色散人,侵删。

下面由golang教程栏目给大家介绍Golang简单的memcache实现方法,希望对需要的朋友有所帮助!

这两天在做项目的过程中遇到了一个访问全局变量的问题场景:编写一个方法,获取id对应的token值,token需要缓存起来(全局变量内存缓存),如果获取不到或者token的时间过期,那么发送http请求到其他端去取,然后缓存起来,然后再返回,那么代码如下:

code.go:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

package person

 

import (

    "time"

)

 

var gAccId2Token map[int]interface{} = make(map[int]interface{})

 

func GetTokenByAccountId(accountId uint, acl string) (map[string]interface{}, error) {

    //get token from cache

    if token, ok := gAccId2Token[accountId]; ok {

        if token != nil {

            now := time.Now().Unix()

            if int(now) < int(token.(map[string]interface{})["expireDate"].(float64)) {

                return token.(map[string]interface{}), nil

            }

        }

    }

 

    token, err := getTokenByHttpUrl(apiUrl)

    if err != nil {

        return map[string]interface{}{}, err

    }

 

    gAccId2Token[accountId] = token

 

    return token.(map[string]interface{}), nil

}

那么问题来了:

1.由于gAccId2Token变量是全局变量,那么会出现同时读写的情况,则会可能出现读写不一致的情况。

2.就本例来看,获取id=2的token,而缓存的token已经过期了,那么就会发送http请求去获取,之后写缓存,假设写缓存的时间很长,而在这段时间内,又恰好有大量请求来获取id=2的token,由于token都过期了,就会出现大量请求http服务端的问题,不仅没有起到获取缓存的目的,又增大了后端的压力,同时又有多个写缓存的操作,而golang的map应该不是原子的,那么大量写内存也可能会造成crash的问题。

因此,我们需要对读写操作进行加锁:

memcache.go:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

package person

 

import (

    "sync"

    "time"

)

 

type memoryCache struct {

    lock  *sync.RWMutex

    items map[interface{}]interface{}

}

 

func (mc *memoryCache) set(key interface{}, value interface{}) error {

    mc.lock.Lock()

    defer mc.lock.Unlock()

    mc.items[key] = value

    return nil

}

 

func (mc *memoryCache) get(key interface{}) interface{} {

    mc.lock.RLock()

    defer mc.lock.RUnlock()

 

    if val, ok := mc.items[key]; ok {

        return val

    }

    return nil

}

 

var gAccId2Token *memoryCache = &memoryCache{

        lock:  new(sync.RWMutex),

        items: make(map[interface{}]interface{}),

    }

 

func GetTokenByAccountId(accountId uint, acl string) (map[string]interface{}, error) {

    //get token from cache

    token := gAccId2Token.get(accountId)

    if token != nil {

        now := time.Now().Unix()

        if int(now) < int(token.(map[string]interface{})["expireDate"].(float64)) {

            return token.(map[string]interface{}), nil

        }

    }

 

    token, err := getTokenByHttpUrl(apiUrl)

    if err != nil {

        return map[string]interface{}{}, err

    }

 

    gAccId2Token.set(accountId, token)

 

    return token.(map[string]interface{}), nil

}

几点说明:

1.为写操作上了全局锁,一旦Lock()之后,其他lock便不能上锁,直到释放锁Unlock()之后才行,也就是说保证写操作的原子性。

2.而为读操作上了读锁,那么可以有多个线程Rlock()对一个区域枷锁,从而保证区域是可读的,直到所有读锁都RUnlock()之后,才可以上写锁。

3.将map的key和value的类型都定义成为interface{}类型,interface{}可以接收任何类型,就像是Java中的Object。

4.interface{}类型转换的方法,value.(type),即将value转换成为type类型,例如:value.(int),value.(map[string]interface{})等等。

以上就是Golang的memcache如何简单实现的详细内容,更多文章请关注木庄网络博客!!

相关阅读 >>

阅读golang 切片(slice)底层源码

golang 开发的 web 有哪些框架?

golang学习笔记for循环语句

go每日一库 [go-rate] 速率限制器

cmd执行golang乱码解决方法

golang语言社区--unity3d学习 第1期 如何学习unity3d

go-carbon 1.3.1 版本发布,新增 diffforhumans() 方法和多语言支持

手撸golang 基本数据结构与算法 插入排序

[译]golang 1.8工具链改进

golang的memcache如何简单实现

更多相关阅读请进入《golang》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...