Go实现web服务器
Go 允许您使用它的标准库函数自己实现一个 web 服务器。在这本书中您第一次看到一个用 Go 实现的 web 服务器应用程序是在第10章,Go 并发-进阶讨论,我们讨论 context
包的时候。
尽管用 Go 实现的 web 服务器能做需要高效安全的事情,但如果您真正需要一个强大的 web 服务器,支持模块,多站点和虚拟主机的话,您最好使用如 Apache 或 Nginx 这样的 web 服务器
这个例子程序命名为 www.go
,由五部分展示。它的第一部分包含期望的 import
声明:
package main
import (
"fmt"
"net/http"
"os"
"time"
)
对于 web 服务器不需要操作 time
包。然而,由于服务器要发送时间和日期给客户端,所以这个例子里需要它。
www.go
的第二段代码如下:
func myHandler(w http.ResponseWriter, r *http.Request) {
fmt.Fprintf(w, "Serving: %s\n", r.URL.Path)
fmt.Printf("Served: %s\n", r.Host)
}
这是该程序的第一个处理函数的实现。一个处理函数取决于使用的配置可以服务一个或多个 URL。
www.go
的第三部分包含如下代码:
func timeHandler(w http.ResponseWriter, r *http.Request) {
t := time.Now().Format(time.RFC1123)
Body := "The current time is:"
fmt.Fprintf(w, "<h1 align=\"center\">%s</h1>", Body)
fmt.Fprintf(w, "<h2 align=\"center\">%s</h2>\n", t)
fmt.Fprintf(w, "Serving: %s\n", r.URL.Path)
fmt.Fprintf(w, "Served time for: %s\n", r.Host)
}
从上面这段 Go 代码,您能看到该程序的第二个处理函数的实现。该函数输出动态内容。
我们的 web 服务器代码的第四部分处理命令行参数和定义支持的 URL:
func main() {
PORT := ":8001"
arguments := os.Args
if len(arguments) == 1 {
fmt.Println("Using default port number: ", PORT)
}else{
PORT = ":" + arguments[1]
}
http.HandleFunc("/time", timeHandler)
http.HandleFunc("/", myHandler)
http.HandleFunc()
函数把一个URL和一个处理函数关联起来。
www.go
程序的最后一段如下:
err := http.ListenAndServe(PORT, nil)
if err != nil {
fmt.Println(err)
return
}
}
您将在 http.ListenAndServe()
函数的帮助下使用期望的端口号启动 web 服务器。
执行 www.go
并连接 web 服务器将生成如下输出:
$ go run www.go
Using default port number: :8001
Served: localhost:8001
Served: localhost:8001
Served time for: localhost:8001
Served: localhost:8001
Served time for: localhost:8001
Served: localhost:8001
Served time for: localhost:8001
Served: localhost:8001
Served: localhost:8001
Served: localhost:8001
尽管这个程序的输出提供了一些方便的信息,但我想您更愿意用您喜欢的浏览器看到真正的输出。下面的截图显示了我们的 web 服务器的 myhandler()
函数的输出,显示在 Google Chrome:
下面的截图显示 www.go
也能生成动态页面。在这个例子,它是注册在 /time
下的 web 页面,显示当前时间和日期:
这里真正有趣的是除了 /time
所有的 URL 都由 myHandler()
函数处理,因为它是 /
的第一个参数,匹配所有没有与另一个 handler 匹配的 URL。