0

问题是这样的:有一个 Web 服务器。我认为在页面加载中使用 goroutine 会很有好处,所以我继续做了:调用 loadPage 函数作为 goroutine。但是,执行此操作时,服务器会停止工作而不会出现错误。它打印一个空白的白页。问题必须出在函数本身——某些东西与 goroutine 有某种冲突。

这些是相关的功能:

func loadPage(w http.ResponseWriter, path string) {
   s := GetFileContent(path)
   w.Header().Add("Content-Type", getHeader(path))
   w.Header().Add("Content-Length", GetContentLength(path))
   fmt.Fprint(w, s)
}
func GetFileContent(path string) string {
   cont, err := ioutil.ReadFile(path)
   e(err)
   aob := len(cont)
   s := string(cont[:aob])
   return s
}


func GetFileContent(path string) string {
   cont, err := ioutil.ReadFile(path)
   e(err)
   aob := len(cont)
   s := string(cont[:aob])
   return s
}

func getHeader(path string) string {
   images := []string{".jpg", ".jpeg", ".gif", ".png"}
   readable := []string{".htm", ".html", ".php", ".asp", ".js", ".css"}
   if ArrayContainsSuffix(images, path) {
      return "image/jpeg"
   }
   if ArrayContainsSuffix(readable, path) {
      return "text/html"
   }
   return "file/downloadable"
}


func ArrayContainsSuffix(arr []string, c string) bool {
   length := len(arr)
   for i := 0; i < length; i++ {
      s := arr[i]
      if strings.HasSuffix(c, s) {
         return true
      }
   }
return false
}
4

2 回答 2

2

发生这种情况的原因是因为您调用“loadPage”的 HandlerFunc 与请求同步调用当您在 go 例程中调用它时,处理程序实际上是立即返回,从而导致立即发送响应。这就是为什么你会得到一个空白页。

您可以在server.go中看到这一点(第 1096 行):

serverHandler{c.server}.ServeHTTP(w, w.req)
if c.hijacked() {
    return
}
w.finishRequest()

ServeHTTP函数调用您的处理程序,并在它返回后立即调用“finishRequest”。因此,只要您的 Handler 函数想要满足请求,它就必须阻塞。

使用 go 例程实际上不会使您的页面更快。正如 Philip 所建议的,在这种情况下,将单个 go 例程与通道同步也对您没有帮助,因为这与根本没有 go 例程相同。

您问题的根源实际上是ioutil.ReadFile,它在发送之前将整个文件缓冲到内存中。

如果要流式传输您需要使用的文件os.Open。您可以使用io.Copy将文件的内容流式传输到浏览器,这将使用分块编码。

看起来像这样:

f, err := os.Open(path)
if err != nil {
    http.Error(w, "Not Found", http.StatusNotFound)
    return
}
n, err := io.Copy(w, f)
if n == 0 && err != nil {
    http.Error(w, "Error", http.StatusInternalServerError)
    return
}

如果由于某种原因您需要在多个 go 例程中工作,请查看sync.WaitGroup. 通道也可以工作。

如果您只想提供文件,还有其他为此优化的选项,例如FileServerServeFile

于 2013-08-02T20:52:08.277 回答
0

在 Go 中的典型 Web 框架实现中,路由处理程序作为 Goroutines 调用。即在某些时候网络框架会说go loadPage(...)

所以如果你从内部 loadPage调用一个 Go 例程,你有两个级别的 Goroutines。

Go 调度器真的很懒惰,如果不是强制的,它不会执行第二级。所以你需要通过同步事件来强制执行它。例如,通过使用渠道或sync包。例子:

func loadPage(w http.ResponseWriter, path string) {
  s := make(chan string)
  go GetFileContent(path, s)
  fmt.Fprint(w, <-s)
}

Go 文档是这样说的:

如果一个 goroutine 的效果必须被另一个 goroutine 观察到,可以使用锁或通道通信等同步机制来建立相对顺序。

为什么这实际上是一件聪明的事情?在大型项目中,您可能会处理大量需要以某种方式有效协调的 Goroutine。那么,如果 Goroutine 的输出无处使用,为什么还要调用它呢?一个有趣的事实:I/O 操作fmt.Printf也会触发同步事件。

于 2013-08-02T20:26:00.843 回答