This topic created in 899 days ago, the information mentioned may be changed or developed.
本人初学 :D ,写了一个简单的 web server , 请教一个简单的问题:
在 main.go 开头有一个全局变量 ( 例如 :var search_result := [] string ) ,用于保存用户最新的搜索结果,请问如果不同的人同时访问这个 server ,搜索不同的内容,都是保存在这个全局变量里面。那么这个全局变量对于这些不同的用户来说,是否分别存在着和这些访问向对应的不同的副本 ?
13 replies • 2023-12-21 13:59:51 +08:00
 |
|
3
yaott2020 Dec 20, 2023 via Android 1
不存在什么副本,但是如果存在多个同时写入可能造成冲突,建议加锁。还有,你要么写 var search_result = []string{},要么写 var search_result []string ,你上面的写法是不正确的
|
 |
|
4
qloog Dec 20, 2023 2
在 Go 语言中,全局变量是在程序生命周期内存在的单一实例。因此,如果你在 main.go 中声明一个全局变量 searchResult ,那么它对所有的请求都是同一个实例。
考虑到多个用户同时访问服务器的情况,这可能导致并发访问的问题。在并发情况下,多个 goroutine ( Go 程序中的轻量级线程)可能会同时尝试读取或写入全局变量,这可能导致竞态条件和数据不一致的问题。
为了解决这个问题,你可以考虑使用互斥锁( Mutex )或者使用 Go 语言中的通道( Channel )来确保对全局变量的访问是安全的。下面是一个简单的例子,演示如何使用互斥锁:
```go package main
import ( "fmt" "net/http" "sync" )
var ( searchResult []string mutex sync.Mutex )
func handleSearch(w http.ResponseWriter, r *http.Request) { // 使用互斥锁确保对全局变量的访问是安全的 mutex.Lock() defer mutex.Unlock()
// 这里可以对 searchResult 进行读取或写入操作 // ...
// 示例:向 searchResult 添加一个搜索结果 searchResult = append(searchResult, "Search result for "+r.URL.Query().Get("query"))
// 返回搜索结果 fmt.Fprintf(w, "Search result added: %s\n", searchResult) }
func main() { http.HandleFunc("/search", handleSearch) http.ListenAndServe(":8080", nil) } ```
在这个例子中,使用了 sync.Mutex 来保护对 searchResult 的并发访问。在 handleSearch 函数中,通过调用 mutex.Lock() 和 mutex.Unlock() 来确保在同一时刻只有一个 goroutine 能够访问 searchResult 。这样可以避免并发访问导致的问题。
|
 |
|
9
jonsmith Dec 20, 2023
包级变量(全局变量)的生命周期和整个程序的运行周期是一致的,Go 主程序( main goroutine )不结束,会一直存在。 每个 web 请求是从主程序 main 创建的子协程 goroutine ,所有请求读写的全局变量是同一块内存,并发写时会有竞争关系,要加锁或使用 channel 。
|
 |
|
10
chengxiao Dec 20, 2023 1
你这个简单且业务量不大的处理方法是 用 map[string][]string 为每个用户 id 设置一个[]string
|
 |
|
11
dyllen Dec 20, 2023
没有副本,所有的用户都是读写同一个 search_result 。
|