PHP的时间

本文详细解析了PHP中时间戳、时区、本地时间及UTC时间的概念与应用技巧,包括如何确保时间输出的准确性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、 Unix时间戳

Unix时间戳是在计算机领域才有的,每一台电脑(服务器)在生产的时候,将GMT/UTC的1970年01月01日00时00分00秒作为起始值进行计算,得到的总秒数就是这个Unix时间戳。至于是GMT还是UTC意义并不大,因为GMT和UTC的1970年01月01日00时00分00秒是一致的,起点一致的情况下,运行的秒数也是一致的。

	//获取时间戳
	time();

二、 时区(Time Zone)

但是上面的time()的表述并不准确,因为我们在实践中经常遇到time()得到的值并不是我们想要的。对应的是,date()函数得到的值,也可能出乎我们意料。

什么是时区呢?也就是以GMT/UTC为参照物的时间偏移。

1. 以GMT为参照物的时区

在传统的教材里,全球被划分为24个时区,首先基于经度,其次按照国家或地区,将每一个地区划分到某一个时区,这样可以避免时间上的混乱。在24时区划分之前,世界上的时间换算并没有准确的参照,比如中国人去英国,只能问当地人现在几点,然后拨自己的表来对。而当时区划分之后,中国人到了英国,只需要拨慢8小时即可。在时区划分之前,英国人跟中国人的时间可能并不是严格的8小时之差。

但为了照顾到同一个国家内时间的统一,大部分国家规定自己属于同一个时区,比如中国,统一规定为东八区,这样中国东部和西部可以采用同一个时间。毕竟没有必要大家一定要在早上6点看到日出,沿海城市5点看,乌鲁木齐9点看,并不影响大家的正常作息。

在php中,提供了大量的地区作为时区切换的标准,例如:

date_default_timezone_set(‘Asia/Shanghai’);echo date(‘Y-m-d H:i:s’); // 获得的是上海所在时区的时间
注意:PRC是中国的地区时标志,并不在Asia中,而是在Others里面找。

2. 以UTC为参照物计算时区

但随着UTC取代GMT成为世界标准时后,时区的计算开始使用UTC作为标准。UTC+8代表东八区,UTC-11代表西十一区。
不过随着精度需求的提升,按大时区计算已经不能满足需求,0.5个时区也被普遍使用,比如UTC+7.5。

在PHP中,我们可以采用这种方式来切换时区。比如:

	date_default_timezone_set('UTC');echo date('Y-m-d H:i:s'); // 获取的是0时区时间
3. 时区给PHP带来的影响

我们上面给出的代码并没有什么实际意义,因为你还不知道为什么要这样去做。实际上,php在使用date函数的时候,会依照所在时区去进行计算。

例如,你的服务器是放在英国的,而服务器的默认配置php.ini中没有规定时区,那么php就会以操作系统默认是时区作为时区进行输出,这就会导致这台服务器上的date()函数输出的时间是以UTC+0作为时区的,如果你的用户在中国,那么网站的访客看到的时间就会少8个小时。

而上面使用date()进行输出的时间,就是我们所说的本地时间。

本地时间,其实是指服务器上的程序输出时间,date函数输出的时间依托Unix时间戳和时区,因此它一定是一个不准确的时间,因为Unix时间戳基本上都是不准确的,但是这个不准确是可以忽略的,严重的是时区的偏差。

4. 造成php输出时间混乱的原因总结起来:

使用默认的date函数的输出值
在保存时间的时候使用调整过时区的时间,而输出时又调整了时区
第一点比较容易理解,比如默认存进去的时候存入的是time(),输出的时候使用date(),time()是没有错的,但是date()在输出的时候,时区和当前访客的时区对不上,从而导致输出内容的错误。

三、如何在PHP中保证输出时间的准确性?

我们想的更多的是如何保证时间的准确性问题。这要从多方面去考虑,输入输出的一致性与非一致性是一个很大的挑战,你需要把握好全局关系。

1. php.ini配置文件中规定时区

从php5.1.0开始,php.ini配置文件中支持设置一个date.timezome的值来规定默认的时区,找到它,并改为:

	//修改php.ini
	date.timezone = PRC

当然,PRC也可以用php官方给出的列表中的其他时区代表值表示。

这种配置的好处是,可以在所有的php代码中生效,坏处是灵活性差,而且大部分主机并不直接支持php.ini配置。

2. ini_set(‘date.timezone’)

在php代码开头,可以使用ini_set函数来临时修改一些php的默认配置,可以:

	//代码修改配置
	ini_set('date.timezone','Asia/Shanghai');

这种方法的好处是比较灵活,需要配置时区的代码里才使用,把这个配置放置在一个共享文件里,可以使所有引用这个文件的php脚本都获得这个配置。坏处是有的主机不支持ini_set。

3. date_default_timezone_set

和ini_set函数一样,date_default_timezone_set函数也可以临时修改php配置。

	//其它函数
	date_default_timezone_set('Asia/Shanghai');
4.自己计算

当你在输出日期的时候,可以考虑自己调整时区,然后进行计算,将计算的结果格式化为日期再输出。首先,我们要搞清楚哪些是可变哪些是不可变。

	//可变:
	date()
	//不可变:
	time();   gmdate()

当你在输出一个日期的时候,如果使用date,就是可变的,但如果使用gmdate()就是不可变的,gmdate()永远把时区当做是UTC+0,即使你通过前面三种方法临时修改了时区,也不会影响gmdate的输出结果,而这个时候,其实你又知道你的访客所在的时区,所以,你可以自己计算一下:

	// 方法1
	date('Y-m-d H:i:s',strtotime(gmdate('Y-m-d H:i:s').' +8 hours'));
	// 方法2(推荐)
	gmdate('Y-m-d H:i:s',time() + 8*3600);

使用上面的两个方法,无论你的服务器处于什么时区,无论你是否使用date_default_timezone_set设置了新的临时时区,都不会影响结果,因为gmdate永远以UTC+0作为参照,根本不会理会你新设置的时区。甚至,你把你的这段代码,从非洲的服务器搬到中国的服务器上,它的结果也还是一样(忽略timestamp的微小误差)。

有一个有趣的现象是,我们可以通过一个动态的数字来控制date()是时区,而无需去设置时区,比如:

	date('Y-m-d H:i:s',time() + n*3600)

其中的n则是时区,东八区就是+8,西五区就是-5。而我们却可以找出这个动态的n值,它和时区时时相关:date(‘Z’)
date(‘Z’)是一个军事级别的应用,它用于计算以秒为单位的时区偏移量,比如东八区,它的值就是8*3600,我们可以在time()的基础上减去这个值,得到一个比当前时间戳少时区偏移量的值,这个值在实际中没有任何意义,它不代表任何时间戳(或者说是当前时间n小时之前的时间戳),但如果我们再对这个时间戳进行date运算时,date会把n时区的偏移量加回来,这样就得到了一个固定的UTC+0的日期时间:

	$gmt_date = date('Y-m-d H:i:s',time() - date('Z'));

它的效果其实和gmdate(‘Y-m-d H:i:s’)相同,但算法上更加复杂。

同样的道理,我们以UTC+0作为基准,增加这个偏移量,反而可以得到我们想要的时区所在的时间:

	$local_time = gmdate('Y-m-d H:i:s',time() + date('Z'));

但这和$local_time = date(‘Y-m-d H:i:s’);没有任何结果上的区别。

选择你合适的时间进行保存
在前面的分析里面,你看到有一种情况比较乱,就是使用了调整时区后的时间进行保存,但是显示的时候,又进行时区调整,这导致显示错误。

推荐的一种时间保存方案,是只保存timestamp,也就是time(),它的值是固定的,不随着时区的调整而改变,即使更换了服务器,它的误差也很小,所以有利于今后将程序分发部署到不同的服务器上面。

而在自己显示的时候,可以确定一个方法,比如上面推荐的方法2作为输出:

		function st_date($format,$timestamp = false) {
		  $timestamp = is_numberic($timestamp) ? $timestamp : time();
		  return gmdate($format,$timestamp + 8*3600);
		}

这样就可以保证这段代码无论在哪里,都可以输出东八区的时间。

本文转载于:
PHP中关于时间(戳)、时区、本地时间、UTC时间等的梳理

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值