Rails 分享 多时区对应的解决案

acidache · 2017年01月24日 · 最后由 guyanbiao 回复于 2017年01月25日 · 1935 次阅读

前段时间,接到了一个 ROR 项目 (之前完全没有接触过 ROR)。。。其中,客户要求提供同时对应多国的时区以及夏令时的功能。(要求很高大上,估计就客户他们国家的人自己玩) 即 服务器比如搭建在中国,客户端 (WEB) 可能是在日本,中国,美国等等,全世界各个角落来访问服务器。

ROR 本身支持了多时间和夏令时的对应,查了各种网站,几乎搜索到的只有 config.time_zoneconfig.active_record.default_timezone 的设定 但是这个的设定都是针对固定某个时区的,无法实现基于用户信息中所含有时区信息来自动切换时区来处理各种请求。

为了实现同时服务不同时区的用户,我们使用了一下的解决思路:
                       local时间                     utc时间
客户端(WEB)  请求   --------->   ROR服务器接收后※1    --------->     各种处理(对DB访问等)
客户端(WEB)  回复   <---------   ROR服务器准备发送※2  <---------     处理结果(从DB中读取数据等)

为了实现上面的思路,修改了以下内容。 1.config.time_zoneconfig.active_record.default_timezone 的默认设置都注释掉。

这样子会使得 DB 直接使用 UTC

2.在application_controller.rb中,追加一个 before_action 动作,来做基于用户的时区来设置 ROR 的时区,来保证了 ROR 服务器回复客户端时会回复 local 时间

 #代码示例
user = User.where(:name => name).first
Time.zone  = user.local

★★★ 需要注意的是,上面的设定只能成功改变通过 ActiveRecord 的 API 来取得 DB 的数据时间时,会按照上面设定的时区自动做转化。

★★★ 直接使用 SQL 语句查询出来的记录会保持数据库里面的 UTC 时间

3.为了给不同的 WEB 端提供统一的时间服务,ROR 服务器在 WEB 端用户登录请求的回复内容中,回复该用户登录时的 local 时间, 然后在 WEB 端,登录后的动作都基于这个收到的 local 时间来执行,来保证 WEB 端与 ROR 服务器的时间的统一性。

上面只是大致提供了一个思路,实际项目时,也还会遇到各种各种的 case,需要具体情况在分析了。

写在最后,作为一个才接触 rails 半年的小白,在 Ruby China 中找到了很多思路和灵感,于是自己也想着分享点自己和小伙伴们这半年的一点心得。 上面的时区转换,找了很久也没有找到合适的例子,因此将目前我们的做法分享出来,应该还有更好的实现方法,我就抛砖引玉吧。

如果有什么不明点和指摘,请留言交流~

需要 登录 后方可回复, 如果你还没有账号请 注册新账号