4

我正在尝试设置一个不缓存静态资产 .css 和 .js 的项目。对于某些人来说,我们似乎遇到了一些内部缓存问题,我希望这能解决这个问题。

我有一个相位监听器,基本上是这个http://turbomanage.wordpress.com/2006/08/08/disable-browser-caching-in-jsf/的略微修改版本

我的课:

package com.ods.common.jsf.phaselistener;

import javax.faces.context.FacesContext;
import javax.faces.event.PhaseEvent;
import javax.faces.event.PhaseId;
import javax.faces.event.PhaseListener;
import javax.servlet.http.HttpServletResponse;

public class CacheControlPhaseListener implements PhaseListener
{
public PhaseId getPhaseId()
{
    return PhaseId.RENDER_RESPONSE;
}

public void afterPhase(PhaseEvent event)
{
}

public void beforePhase(PhaseEvent event)
{
    FacesContext facesContext = event.getFacesContext();
    HttpServletResponse response = (HttpServletResponse) facesContext
            .getExternalContext().getResponse();
    response.setHeader("Cache-control", "no-cache"); // HTTP 1.1
    response.setHeader("Cache-control", "no-store"); // HTTP 1.1
    response.setHeader("Cache-control", "must-revalidate"); // HTTP 1.1
    // response.setHeader("Pragma","no-cache"); //HTTP 1.0
    response.setHeader("Allow", "GET"); // Allowing GET Method only
    response.setHeader("Allow", "POST");// Allowing POST Method only
    response.setDateHeader("Expires", -1); // prevent caching at the proxy server

            /*what I've added*/
    response.setHeader("Pragma", "no-cache");
    response.setHeader("Content-type", "x-javascript");
            //below are the content types I'm seeing for the css/js assets via Firebug
    response.setHeader("Content-type", "application/x-javascript");
    response.setHeader("Content-type", "text/css");
}
}

我对 faces-config.xml 的补充:

<lifecycle>
    <phase-listener id="nocache">com.ods.common.jsf.phaselistener.CacheControlPhaseListener</phase-listener>
</lifecycle>

我的 .xhtml 页面过去有一个 expires 标头:

Expires Thu, 01 Jan 1970 00:00:00 GMT

所以这似乎有点工作......我假设这个日期来自 Expires -1(将其设置为 unix 时代)。

如您所见,我尝试为一些 javascript 和 css 设置适当的内容类型标头,但这些资产的到期日期是未来一周。

有人有想法么?另外,我是前端开发人员,而不是后端 Java 人。我可以乱用 Java,但我绝对不是 Java 开发人员。这也是我在 JSF 的第一份工作,所以尽可能地低调一点就好:)

4

1 回答 1

13

随着setHeader()您覆盖任何先前设置的标题。而是使用addHeader(),或者只是将所有值以逗号分隔作为标题值。这是完整的集合:

response.setHeader("Cache-Control", "no-cache, no-store, must-revalidate"); // HTTP 1.1.
response.setHeader("Pragma", "no-cache"); // HTTP 1.0.
response.setDateHeader("Expires", 0); // Proxies.

你的另一个错误是 aPhaseListener不是最好的地方。它仅在 JSF 页面请求上调用,而不是在由 web 浏览器独立调用的静态资源请求上调用。换句话说,只有 JSF 页面本身禁用了缓存,但所有<script>, <link>,<img>等都会生成不会调用的新请求,PhaseListener因为这些不是 JSF 页面。

而是使用Filter.

@WebFilter("/*")
public class NoCacheFilter implements Filter {

    @Override
    public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
        HttpServletResponse response = (HttpServletResponse) res;
        response.setHeader("Cache-Control", "no-cache, no-store, must-revalidate"); // HTTP 1.1.
        response.setHeader("Pragma", "no-cache"); // HTTP 1.0.
        response.setDateHeader("Expires", 0); // Proxies.
        chain.doFilter(req, res);
    }

    // ... (just keep init() and destroy() NO-OP)
}

如果您以 Servlet 3.0 容器(Tomcat 7、Glassfish 3 等)为目标,则不需要web.xml(或)注册。faces-config.xml@WebFilter("/*")自动注册它并将其映射到一个 URL 模式/*,从而覆盖所有请求。

也可以看看:


与具体问题无关,完全禁用静态资产缓存并不是最好的主意。它不必要地消耗网络带宽。而是寻找不同的解决方案,例如在查询字符串中包含服务器启动时间戳。

例如

<script src="foo.js?#{startup.time}"></script>

之内faces-config.xml

<managed-bean>
    <managed-bean-name>startup</managed-bean-name>
    <managed-bean-class>java.util.Date</managed-bean-class>
    <managed-bean-scope>application</managed-bean-scope>
</managed-bean>

此示例将强制浏览器在服务器重新启动时重新加载资产。

于 2012-06-01T17:11:03.560 回答