首页 > 文章列表 > 保证Golang中变量和映射的只读性的方法

保证Golang中变量和映射的只读性的方法

193 2024-02-12
问题内容

我希望我的程序能够访问全局只读正则表达式和映射。这是我的想法:

var myregex *regexp.regexp
var mymap map[string]string

func init() {
    myregex = regexp.mustcompile("blah")
    mymap = map[string]string{"blah": "blah"}
}

或者我可以做

type myStruct struct {
    //  already have bunch of other struct fields
    myRegex    *regexp.Regexp
    myMap map[string]string
}


func Initialize() *myStruct {
    m := myStruct {
        // bunch of other stuff
        myRegex: regexp.MustCompile("blah")
        myMap: map[string]string{"blah": "blah"}
}

但是我如何确保这些变量不会被我的软件的任何其他部分写入?


正确答案


只要您处理指针(地图无论如何都是指针),您就永远无法确保您的地图或正则表达式是只读的。

(好吧,除非你每次都用函数复制值并返回一个新指针......但我不确定这就是你想要实现的:)

如果我以您的示例为例,并添加一个简单的主代码:

package main

import (
    "fmt"
    "regexp"
)

var myregex *regexp.regexp = regexp.mustcompile("blah")
var mymap map[string]string

func init() {
    myregex = regexp.mustcompile("blah")
    mymap = map[string]string{"blah": "blah"}
}

type mystruct struct {
    //  already have bunch of other struct fields
    myregex *regexp.regexp
    mymap   map[string]string
}

func initialize() mystruct {
    return mystruct{
        // bunch of other stuff
        myregex: regexp.mustcompile("blah"),
        mymap:   map[string]string{"blah": "blah"},
    }
}

func getmap() map[string]string {
    return mymap
}

func main() {
    fmt.println(mymap)
    mymap["blah2"] = "blah2"
    fmt.println(mymap)
    fmt.println(getmap())

    m := initialize()

    fmt.println(m.mymap)
    m.mymap["test"] = "test"
    fmt.println(m.mymap)
}

你看我可以修改地图:

❯ ./main
map[blah:blah]
map[blah:blah blah2:blah2]
map[blah:blah blah2:blah2]
map[blah:blah]
map[blah:blah test:test]

正则表达式完全相同。

如果您确实想确保您的正则表达式和映射永远不会被另一段代码错误地更新,有几种解决方案;其中大多数包括将只读变量移动到自己的包中,并且从不直接访问它们。例如这样的事情

package mapreadonly

type ReadOnlyMap struct {
    m map[string]string
}

func (elem ReadOnlyMap) Get(key string) (string, bool) {
    value, ok := elem.m[key]
    return value, ok
}

var Map1 ReadOnlyMap = ReadOnlyMap{
    m: map[string]string{
        "blah": "blah",
    },
}

然后将此包导入到需要它的其他文件中。

但如上所述,您的问题缺乏一些背景信息,无法确保答案符合您的预期。