草庐IT

php - 如何存储重复日期牢记夏令时

coder 2023-06-10 原文

我正在我的数据库中存储事件。我有“开始”和“结束”日期时间、“tickets_start”和“tickets_end”(用于门票销售实际开始/结束的时间 - 而不是实际事件的开始/结束)。

到目前为止,我已经构建了一些方法来完成所有有趣的事情,比如在保存之前将日期/时间转换为 GMT,然后返回到它们各自的时区进行显示。

我将时区存储在 varchar 字段中,其值类似于“America/New_York”。

但是 - 现在我需要开始处理用户是否希望允许重复事件。我以前做过,没什么大不了的,但从来没有跨越多个时区。

起初,我认为这没什么大不了的,但后来意识到 - 如果最初的开始日期是 7 月(例如),并且它每个月重复一年,在某个时候,夏令时将使它这样格林威治标准时间的转换将改变不同的时间。一个月,在转换 12:00 时,它会将其更改为 -5,下一次,由于 DST,它会将其更改为 -4。

我目前的想法是,我将存储一个“dst”tinyint(1) 以表示是否在 DST 期间输入了开始/结束日期,然后在必要时制定一种方法将时间更改一小时。

但是 - 我想我会在这里问,希望可能有一个“正常”或一个我没有想到的简单的东西。

(cakephp 2.4.x)

最佳答案

首先,请注意,在现代术语中,您应该说 UTC 而不是 GMT。它们大部分是等效的,只是 UTC 的定义更精确。保留术语 GMT 以指代英国在冬季月份生效的时区部分,偏移 UTC+0。

现在回答你的问题。 UTC 不一定是存储所有日期和时间值的最佳方式。它对过去的事件或 future 的绝对事件特别有效,但对 future 的局部事件——尤其是 future 的重复发生的事件——效果不佳。

我写过这个 on another answer最近,并且是本地时间比 UTC 更有意义的少数异常(exception)情况之一。主要论点是“闹钟问题”。如果您按 UTC 设置闹钟,您将在夏令时转换当天提前或晚起一小时。这就是为什么大多数人按本地时间设置闹钟的原因。

当然,如果您要处理来自世界各地的数据,则不能只存储本地时间。您应该存储一些不同的东西:

  • 重复事件的本地时间,例如“08:00”
  • 表示本地时间的时区,例如“America/New_York”
  • 重复模式,无论何种格式对您的应用程序有意义,例如每天、每两周或每月的第三个星期四等。
  • 下一个直接的 UTC 日期和时间等价物,尽你所能。
  • 也许,但不总是, future 事件 UTC 日期和时间的列表,将某个预定义的时间段转换到 future (根据您的需要,可能是一周,可能是 6 个月,可能是一年或两年)。

  • 对于最后两个,请了解,如果负责该时区的政府决定更改任何内容,则任何本地日期/时间的 UTC 等效值都可能更改。由于每年都会有多个时区数据库更新,你会想要有一个计划到 subscribe to announcements of updatesupdate your timezone database经常。每当您更新时区数据时,您都需要重新计算所有 future 事件的 UTC 等效时间。

    如果您打算显示跨越多个时区的任何类型的事件列表,则拥有 UTC 等效项很重要。这些是您将查询以构建该列表的值。

    要考虑的另一点是,如果事件被安排在 DST 回退转换期间发生的本地时间,您将必须决定该事件是发生在第一个实例(通常)还是第二个实例(有时)上,或两者(很少),并在您的应用程序中构建一种机制,以确保事件不会触发两次,除非您想要它。

    如果您正在寻找一个简单的答案 - 对不起,但没有。跨时区安排 future 的事件是一项复杂的任务。

    替代方法

    我已经让一些人向我展示了一种他们确实使用 UTC 时间进行调度的技术,即他们选择本地时间的开始日期,将其转换为 UTC 进行存储,并存储时区 ID。然后在运行时,他们应用时区将原始 UTC 时间转换回本地时间,然后使用该本地时间来计算其他重复次数,就好像它是上面最初存储的时间一样。

    虽然这种技术有效,但缺点是:
  • 如果在第一个实例运行之前有一个时区更新改变了本地时间,它会导致整个计划中断。这可以通过为“第一个”实例选择过去的时间来缓解,这样第二个实例实际上是第一个实例。
  • 如果时间确实是应该跟随用户的“ float 时间”(例如在手机上的闹钟中),您仍然必须存储它最初创建时所在区域的时区信息 - 即使这不是您想要运行的区域。
  • 它增加了额外的复杂性,但没有任何好处。如果您可能有一个仅支持 UTC 的调度程序,而您正试图将时区支持改进到其中的情况下,我会保留此技术。
  • 关于php - 如何存储重复日期牢记夏令时,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19626177/

    有关php - 如何存储重复日期牢记夏令时的更多相关文章

    1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

      我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

    2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    4. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

      给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

    5. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

      我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

    6. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    7. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

      我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

    8. ruby - 如何指定 Rack 处理程序 - 2

      Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

    9. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

      在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

    10. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

      我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

    随机推荐