http.FileServer 方法属于标准库 net/http,返回一个使用 FileSystem 接口 root 提供文件访问服务的 HTTP 处理器。可以方便的实现静态文件服务器。
http.ListenAndServe(":8080", http.FileServer(http.Dir("/files/path")))
访问 http://127.0.0.1:8080,即可看到类似 Nginx 中 autoindex 目录浏览功能。
源码解析
我们现在开始将上述的那仅有的一行代码进行剖析,看看到底是如何实现的。源码中英文注释也比较详细,可以参考。
我们先看 http.Dir(),再看 http.FileServer(),而 http.ListenAndServe()
监听 TCP 端口并提供路由服务,此处不赘述。
http.Dir()
从以下源码我们可以看出,type Dir string 实现了 type FileSystem interface 的接口函数 Open,http.Dir("/") 实际返回的是 http.Dir 类型,将字符串路径转换成文件系统。
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
|
// 所属文件: src /net/http/fs .go, 26-87行
type Dir string
func (d Dir) Open(name string) (File, error) {
// ...
}
type FileSystem interface {
Open(name string) (File, error)
}
http.FileServer()
http.FileServer() 方法返回的是 fileHandler 实例,而 fileHandler 结构体实现了 Handler 接口的方法 ServeHTTP()。ServeHTTP 方法内的核心是 serveFile() 方法。
// 所属文件: src /net/http/fs .go, 690-716行
type fileHandler struct {
root FileSystem
}
func FileServer(root FileSystem) Handler {
return &fileHandler{root}
}
func (f *fileHandler) ServeHTTP(w ResponseWriter, r *Request) {
upath := r.URL.Path
if !strings.HasPrefix(upath, "/" ) {
upath = "/" + upath
r.URL.Path = upath
}
serveFile(w, r, f.root, path.Clean(upath), true )
}
// 所属文件: src /net/http/server .go, 82行
type Handler interface {
ServeHTTP(ResponseWriter, *Request)
}
|
serveFile()
方法判断,如果访问路径是目录,则列出目录内容,如果是文件则使用 serveContent()
方法输出文件内容。serveContent()
方法则是个读取文件内容并输出的方法,此处不再贴代码。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
// 所属文件: src /net/http/fs .go, 540行
// name is '/' -separated, not filepath.Separator.
func serveFile(w ResponseWriter, r *Request, fs FileSystem, name string, redirect bool) {
// 中间代码已省略
if d.IsDir() {
if checkIfModifiedSince(r, d.ModTime()) == condFalse {
writeNotModified(w)
return
}
w.Header().Set( "Last-Modified" , d.ModTime().UTC().Format(TimeFormat))
dirList(w, r, f)
return
}
// serveContent will check modification time
sizeFunc := func() (int64, error) { return d.Size(), nil }
serveContent(w, r, d.Name(), d.ModTime(), sizeFunc, f)
}
|
支持子目录路径
http.StripPrefix()
方法配合 http.Handle()
或 http.HandleFunc()
可以实现带路由前缀的文件服务。
1
2
3
4
5
6
7
8
9
10
11
12
|
package main
import (
"net/http"
"fmt"
)
func main() {
http.Handle( "/tmpfiles/" , http.StripPrefix( "/tmpfiles/" , http.FileServer(http.Dir( "/tmp" ))))
err := http.ListenAndServe( ":8080" , nil)
if err != nil {
fmt .Println(err)
}
}
|
总结
以上所述是小编给大家介绍的解析Go 标准库 http.FileServer 实现静态文件服务,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对服务器之家网站的支持!
原文链接:https://www.cnblogs.com/shockerli/archive/2018/08/21/golang-pkg-http-file-server.html