您的位置:

Last-Modified深入浅出

一、基础概念

Last-Modified是HTTP的一个头部信息,指示一个被请求资源(如HTML、CSS、JavaScript等)的最后修改时间。它表示的是UTC时间的日期和时间字符串,以RFC 7231规范中的日期时间格式来呈现。例如:

Last-Modified: Wed, 18 Nov 2020 16:14:31 GMT

当客户端请求某个资源时,会将上次请求该资源时得到的Last-Modified值通过If-Modified-Since请求头部发送回服务器。如果该资源在此时间后没有被修改过,服务器会返回HTTP 304“Not Modified”状态码,否则返回200状态码和新的资源内容。

另外需要注意的是,如果服务器无法获取被请求资源的最后修改时间,或者该资源被修改的时间不可知,则该头部信息的值应设为一个固定的时间标记(如“0000-01-01T00:00:00Z”)。同时,当一个资源被创建时,也可以通过设置Last-Modified值为创建时间来表示该资源暂时为最新版本。

二、缓存策略优化

利用Last-Modified头部信息可以实现缓存策略的优化。在第一次请求某个资源时,服务器会返回资源的Last-Modified值,并在响应头部加入Cache-Control、Expires等缓存相关的头部信息,告知客户端应该缓存该资源的有效期限。当下一次请求该资源时,客户端会发送If-Modified-Since请求头部信息以告知服务器缓存情况,如果服务器返回304状态码,表示客户端的缓存仍然是最新的,并且该资源的有效期限还未到期,则客户端不需要再次请求该资源,因为缓存中的版本已经是最新的了。

同时,服务器也可以结合Etag头部信息来进一步细化缓存策略,当资源被改动后,服务器会更改资源的Etag值,并在下次请求时一并返回,如果客户端的Etag值与服务端的不一致,则表示缓存已失效,需要重新获取最新的资源。

三、服务端支持和使用注意事项

为了使服务器支持Last-Modified头部信息,需要在服务端编写相应的代码逻辑以在响应头部中加入Last-Modified头部信息,并处理客户端的If-Modified-Since请求头部信息。以下是一个简单的例子:

import (
    "net/http"
    "time"
)

func handler(w http.ResponseWriter, r *http.Request) {
    // 设置Last-Modified信息
    modTime := time.Date(2020, time.November, 18, 16, 14, 31, 0, time.UTC)
    w.Header().Set("Last-Modified", modTime.UTC().Format(http.TimeFormat))

    // 处理If-Modified-Since信息
    if modHeader := r.Header.Get("If-Modified-Since"); modHeader != "" {
        modSince, err := time.Parse(http.TimeFormat, modHeader)
        if err == nil && !modTime.After(modSince) {
            w.WriteHeader(http.StatusNotModified)
            return
        }
    }

    // 返回资源内容
    http.ServeFile(w, r, "path/to/resource")
}

需要注意的是,由于Last-Modified值是按UTC时间计算的,因此在响应头部中必须将时间转换为UTC格式。同时,如果服务器无法获取被请求资源的修改时间,可以忽略掉If-Modified-Since请求头的判断,直接返回最新的版本即可。

四、总结

Last-Modified头部信息是HTTP协议中用于表示资源修改时间的一种标准方式,可以实现缓存、更新等优化策略,提升HTTP的性能和可靠性。在实际开发中,我们需要充分理解其原理,合理应用其优点,并注意一些问题和注意事项,以确保系统的正确性和可靠性。