让tomcat使用强制ETag参数解除浏览器对静态文件的缓存

Etag在HTTP1.1中有介绍,主要的作用就是在(css file, image, javascript file)文件

请求返回的http头加入ETag参数,Etag有服务器端生成,并且随着文件的改变而改变,这样浏览器端就会只重新请求获取 Etag发生变化的文件,减少浏览器端数据的流量,加快浏览器的反应速度,重要的是减轻服务器端的压力,所以服务器端Etag的实现就比较重要了。

ETag有两种,一种是弱类型的(Weak ETag),一种是强类型的(Strong ETag),强类型格式为ETag=”文件长-最后修改时间”,弱类型是在前面加上W/如:W/”文件长-最后修改时间”,在浏览器中监控如下:

1.第一次请求一个静态文件返回:

HTTP/1.1 200 Ok
ETag: W/"1837-1431071955000"

2.第二次请求:

If-None-Match: W/"1837-1431071955000" 
If-Modified-Since: Fri, 08 May 2015 07:59:15 GMT

返回:

HTTP/1.1 304 Not Modified 
ETag: W/"1837-1431071955000" 
Date: Fri, 28 Oct 2016 02:47:04 GMT 

默认情况下使用弱类型的ETag浏览器会忽略设个机制,而且tomcat默认就是使用的这种机制,所以当你更新一个静态文件比如脚本啊或者一个样式文件,发现浏览器里面还是原来的文件,必须清理缓存或者在文件后面加上一个参数才能自动更新浏览器的缓存,而且tomcat也没有可配置的地方能够配置tomcat使用请类型的ETag,下面提供一个方法来修改这个机制,就是继承类org.apache.naming.resources.FileDirContext,重写doGetAttributes,在返回的属性中加入强类型ETag就可以了,代码如下:

package cn.fullstacks.tomcat;

import java.io.File;

import javax.naming.NamingException;
import javax.naming.directory.Attributes;

/**
 * 在http返回头中返回强类型ETag代替tomcat的弱类型ETag
 * 在context.xml中配置
 * <Resources className="cn.fullstacks.tomcat.ETagFileDirContext" />
 * @author Administrator
 *
 */
public class ETagFileDirContext extends org.apache.naming.resources.FileDirContext {

	@Override
	protected Attributes doGetAttributes(String name, String[] attrIds)
			throws NamingException {
		File file = file(name);

		if (file == null) {
			return null;
		}
		FileResourceAttributes fra = new FileResourceAttributes(file);
		fra.setETag("\"" + fra.getContentLength() + "-" +fra.getLastModified() + "\"");
		return fra;
	}

}

在tomcat的配置文件context.xml中加入配置如下:

<context>
    <Resources className="cn.fullstacks.tomcat.ETagFileDirContext" />
...
</context>

这样请求返回的ETag如下:

HTTP/1.1 200 Ok
ETag: "7664-1477637145682"

这样每次更新静态文件只要最后修改日期变更,那么浏览器必定会重新加载,这样就解决了浏览器静态文件缓存更新的问题。



php cookie用法之 setcookie()函数

setcookie(PHP 3, PHP 4, PHP 5)
setcookie — 发送一个 cookie 信息

说明bool setcookie ( string name [, string value [, int expire [, string path [, string domain [, bool secure]]]]] )

setcookie() 定义一个和其余的 HTTP 标头一起发送的 cookie。和其它标头一样,cookie 必须在脚本的任何其它输出之前发送(这是协议限制)。

这需要将本函数的调用放到任何输出之前,包括 <html> 和 <head> 标签以及任何空格。如果在调用 setcookie() 之前有任何输出,本函数将失败并返回 FALSE。

如果 setcookie() 函数成功运行,将返回 TRUE。这并不说明用户是否接受了 cookie。

注: 自 PHP 4 起,可以用输出缓存来在调用本函数前输出内容,代价是把所有向浏览器的输出都缓存在服务器,直到下命令发送它们。

可以在代码中使用 ob_start() 及 ob_end_flush() 来实现这样的功能,或者通过修改 php.ini 中的 output_buffering 配置选项来实现,也可以通过修改服务器配置文件来实现。

除了 name 外,其它所有参数都是可选的。可以用空字符串(””)替换某参数以跳过该参数。因为参数 expire 是整型,不能用空字符串掉过,可以用零(0)来代替 。

下面的表格对 setcookie() 的每一个参数都进行了解释。可以对照 Netscape cookie 规范以了解 setcookie() 的每一个参数的细节以及通过阅读 RFC 2965 了解 HTTP cookie 的工作方式。

表格 1. setcookie() 参数详解
[参数说明举例]

name Cookie 的名字  使用 $_COOKIE[‘cookiename’] 调用名为 cookiename 的 cookie
value  Cookie 的值 此值保存在客户端,不要用来保存敏感数据。    假定 name 是 ‘cookiename’,可以通过 $_COOKIE[‘cookiename’] 取得其值
expire Cookie 过期的时间 Cookie 过期的时间。这是个 Unix 时间戳,即从 Unix 纪元开始的秒数。换而言之,通常用 time() 函数再加上秒数来设定
cookie 的失效期。或者用 mktime()来实现。    time()+60*60*24*30 将设定 cookie 30
天后失效。如果未设定,cookie 将会在会话结束后(一般是浏览器关闭)失效
path Cookie 在服务器端的有效路径  如果该参数设为 ‘/’ 的话,cookie 就在整个 domain 内有效,如果设为 ‘/foo/’,cookie 就只在 domain 下的 /foo/ 目录及其子目录内有效,例如 /foo/bar/。默认值为设定 cookie 的当前目录
domain  该 cookie 有效的域名 要使 cookie 能在如 example.com 域名下的所有子域都有效的话,该参数应该设为 ‘.example.com’。虽然 . 并不必须的,但加上它会兼容更多的浏览器。如果该参数设为 www.example.com 的话,就只在 www 子域内有效
secure  指明 cookie 是否仅通过安全的 HTTPS 连接传送 指明 cookie 是否仅通过安全的 HTTPS 连接传送。当设成 TRUE 时,cookie 仅在安全的连接中被设置。默认值为 FALSE。    0 或 1

当 cookie 被设置后,便可以在其它页面通过 $_COOKIE 或 $HTTP_COOKIE_VARS 数组取得其值。

需要注意的是,autoglobals 的 $_COOKIE 形式适用于 PHP 4.1.0 或更高版本。而 $HTTP_COOKIE_VARS 则从 PHP 3 起就可以使用。Cookie 的值也会被保存到 $_REQUEST 数组中。

注:

如果 PHP 的选项 register_globals 被设为 on 的话,cookie 的值仍然会被斌到变量内。在下面的例子中,$TestCookie 会被注册,但是仍然推荐使用 $_COOKIE 数组。

常见缺陷:

    Cookies 不会在设置它的本页生效,要测试一个 cookie 是否被成功的设定,可以在其到期之前通过另外一个页面来访问其值。过期时间是通过参数 expire 来设置的。可以简单地使用 print_r($_COOKIE); 来调试现有的 cookies。
    Cookie 必须用和设定时的同样的参数才能删除。如果其值一个空字符串,或者是 FALSE,并且其它的参数都和前一次调用 setcookie 时相同,那么所指定名称的 cookie 将会在远程客户端被删除。
    由于把 cookie 的值设为 FALSE 会使客户端尝试删除这个 cookie,所以要在 cookie 上保存 TRUE 或 FALSE 时不应该直接使用 boolean 值,而应该用 0 来表示 FALSE,用 1 来表示 TRUE
    可以把 cookie 的名称设置成一个数组,但是数组 cookie 中的每个元素的值将会被单独保存在用户的系统中。考虑使用 explode() 函数用多个名称和值设定一个 cookie。不推荐将 serialize() 用于此目的,因为它可能会导致一个安全漏洞。

在 PHP 3 中,在同一个 PHP 脚本中多次使用 setcookie() 来设置 cookie,将会按照倒序的方式来分别执行,如果想要在插入另外一个 cookie 之前删除一个 cookie,要把插入放到删除之前。自 PHP 4 起,多次调用 setcookie() 则是按照顺序来执行的。
下面一些例子说明了如何发送 cookie:
例子 1. setcookie() 发送例子
$value = ‘something from somewhere’;
setcookie(“TestCookie”, $value);
setcookie(“TestCookie”, $value,time()+3600); /* expire in 1 hour */
setcookie(“TestCookie”, $value,time()+3600, “/~rasmus/”, “.utoronto.ca”, 1);

注意 cookie 中值的部分在发送的时候会被自动用 urlencode 编码并在接收到的时候被自动解码并把值赋给与自己同名的 cookie 变量。如果不想这样并且在使用 PHP 5 的话,可以用 setrawcookie() 来代替。下面这个简单的例子可以得到刚才所设定的 cookie 的值:
<?php
// 输出单独的 cookie
echo $_COOKIE[“TestCookie”;
echo $HTTP_COOKIE_VARS[“TestCookie”;

// 另一个调试的方法就是输出所有的 cookie
print_r($_COOKIE);
?>

要删除 cookie 需要确保它的失效期是在过去,才能触发浏览器的删除机制。下面的例子说明了如何删除刚才设置的 cookie:
例子 2. setcookie() 删除例子
// 将过期时间设为一小时前
setcookie(“TestCookie”, “”, time() – 3600);
setcookie(“TestCookie”, “”, time() – 3600, “/~rasmus/”, “.utoronto.ca”, 1);

也可以通过在 cookie 名称中使用数组符号来设定数组 cookie,可以设定多个 cookie 作为数组单元,在脚本提取 cookie 时所有的值都放在一个数组种:
例子 3. setcookie() 中使用数组的例子
<?php
// 设定 cookie
setcookie(“cookie[three]”, “cookiethree”);
setcookie(“cookie[two]”, “cookietwo”);
setcookie(“cookie[one]”, “cookieone”);

// 刷新页面后,显示出来
if (isset($_COOKIE[‘cookie’)) {
foreach ($_COOKIE[‘cookie’] as $name => $value) {
echo “$name : $value <br />\n”;
}
}
?>
上例将输出: three : cookiethreetwo : cookietwoone : cookieone

PHP编程效率的20个要点

用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则
不会,注意:只有echo能这么做,它是一种可以把多个字符串当作参数的“函数”(译注:PHP手册中说echo是语言结构,不是真正的函数,故把函数加
上了双引号)。

1、如果能将类的方法定义成static,就尽量定义成static,它的速度会提升将近4倍。

2、$row[’id’] 的速度是$row[id]的7倍。

3、echo 比 print 快,并且使用echo的多重参数(译注:指用逗号而不是句点)代替字符串连接,比如echo $str1,$str2。

4、在执行for循环之前确定最大循环数,不要每循环一次都计算最大值,最好运用foreach代替。

5、注销那些不用的变量尤其是大数组,以便释放内存。

6、尽量避免使用__get,__set,__autoload。

7、require_once()代价昂贵。

8、include文件时尽量使用绝对路径,因为它避免了PHP去include_path里查找文件的速度,解析操作系统路径所需的时间会更少。

9、如果你想知道脚本开始执行(译注:即服务器端收到客户端请求)的时刻,使用$_SERVER[‘REQUEST_TIME’]要好于time()。

10、函数代替正则表达式完成相同功能。

11、str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。

12、如果一个字符串替换函数,可接受数组或字符作为参数,并且参数长度不太长,那么可以考虑额外写一段替换代码,使得每次传递参数是一个字符,而不是只写一行代码接受数组作为查询和替换的参数。

13、使用选择分支语句(译注:即switch case)好于使用多个if,else if语句。

14、用@屏蔽错误消息的做法非常低效,极其低效。

15、打开apache的mod_deflate模块,可以提高网页的浏览速度。

16、数据库连接当使用完毕时应关掉,不要用长连接。

17、错误消息代价昂贵。

18、在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。

19、递增一个全局变量要比递增一个局部变量慢2倍。

20、递增一个对象属性(如:$this->prop++)要比递增一个局部变量慢3倍。