草庐IT

java - 在 Web 应用程序中处理时区

coder 2024-03-05 原文

在我们的 Web 应用程序中,我们需要显示并输入
不同时区不同国家的日期时间信息。现在,我们正在为每个国家/地区维护单独的 Web 服务器和单独的数据库(oracle 11g)。

我们计划将所有内容合并为一个具有单个数据库的门户(oracle 11g)。
此门户应在用户本地时区捕获/显示日期和时间。

到目前为止,我已经搜索了这个,我得到了以下建议。

1) 将 Web 服务器和数据库服务器的时区设置为 UTC,并在获取数据(数据和时间)时转换为用户本地时区。

如果你建议这种方法,那么请澄清以下具体
问题。

  • 大多数时候我们单独捕获日期,是否需要
    总是捕获日期和时间以及时区?
  • 在存储我们需要转换用户本地时间的日期和时间时
    在 javascript/java/oracle 中将区域转换为 UTC?
  • 在获取我们需要将 UTC 转换为用户的日期和时间时
    本地时区查询本身/java/java 脚本?
  • 我们有许多地方可以根据日期列显示报告,例如
    今天/当前月份/日期范围。我们如何处理这个(输入 - 用户
    本地时区 - UTC 数据库)?
  • 我们必须为日期字段使用哪种数据类型
    (日期/时间戳/带时区的时间戳/带本地时间的时间戳
    区)?

  • 2) 在用户本地时区和 UTC 中捕获日期和时间。存储为单独的列,用户本地时区将用于显示目的,UTC 将用于业务逻辑。

    如果你建议这种方法,那么请澄清以下具体
    问题。
  • 存储用户本地时区和 UTC 是常见的做法吗?
  • 在获取报告以显示时我必须检查哪一列
    基于日期列,例如今天/当月/日期范围?
  • 我们必须为日期列使用哪种数据类型
    (日期/时间戳/带时区的时间戳/带本地时间的时间戳
    区)?

  • 提前致谢

    最佳答案

    阅读问题 Daylight saving time and time zone best practices .你的基本上是重复的。

    UTC 中的服务器

    是的,通常服务器应该将它们的操作系统设置为 UTC作为时区,如果未提供,请使用 GMTReykjavík Iceland time zone .您的 Java 实现可能会将此设置作为其自己的当前默认时区。

    指定时区

    但不要依赖于设置为 UTC 的时区。系统管理员可以更改它。 JVM 中任何应用程序的任何线程中的任何 Java 代码都可以在运行时通过调用 TimeZone.setDefault 来更改 JVM 当前的默认时区。 .因此,通过在 Java 代码中传递可选参数,养成始终指定所需/预期时区的习惯。

    我认为任何日期时间框架都会使时区成为可选的设计缺陷。可选会造成无穷无尽的困惑,因为程序员和其他人一样,除非得到提示,否则会无意识地根据自己的个人时区进行思考。因此,在日期时间工作中,经常没有注意这个问题。添加JVM默认值变化的问题。顺便说一下, Locale 也是如此,同样的问题,应始终明确指定。

    世界标准时间

    您的业​​务逻辑、数据存储和数据交换应该几乎总是在 UTC 中完成。 .几乎每个数据库都具有调整任何输入到 UTC 并以 UTC 存储的功能。

    向用户显示日期时间时,调整到预期的时区。序列化日期时间值时,请使用 ISO 8601字符串格式。见 the Answer由 VickyArora 专门为 Oracle 编写(我是 Postgres 人)。请务必仔细阅读文档,并通过试验来充分理解数据库的行为。 SQL 规范在这方面没有详细说明,行为差异很大。

    sql语句

    请记住,在使用 Java 和 JDBC 时,您将使用 java.sql.Timestamp 和相关的数据类型。它们总是自动地使用 UTC。将来希望看到 JDBC 驱动程序更新以直接使用 Java 8 及更高版本中内置的 java.time 框架中定义的新数据类型。

    时间

    旧类已被 java.time 过时. 学习使用java.time 同时避免旧的 java.util.Date/.Calendar 并使您的编程生活更加愉快。

    直到您的 JDBC driver更新后,您可以使用 java.time 中内置的转换便利方法。参见接下来的示例,其中 Instant 是 UTC 时间和 ZonedDateTime 是 Instant 调整为 time zone .

    Instant instant = myJavaSqlTimestamp.toInstant();
    ZoneId zoneId = ZoneId.of( "America/Montreal" );
    ZonedDateTime zdt = ZonedDateTime.ofInstant( instant , zoneId );
    

    去另一个方向。
    java.sql.Timestamp myJavaSqlTimestamp = java.sql.Timestamp.from( zdt.toInstant() );
    

    如果您需要原始时区,请将其存储

    如果您的业务需求认为原始输入数据的时区很重要,需要记住,则将其显式存储为数据库表中的单独列。您可以使用 offset-from-UTC ,但这并不能提供完整的信息。时区是一个偏移量加上一组用于过去、现在和将来处理异常的规则,例如 Daylight Saving Time .所以一个 proper time zone name最合适如 America/Montreal .

    仅日期不明确

    你说你收集了许多仅限日期的值,没有时间和时区。 java.time 中的类是 LocalDate .与 LocalTime 一样和 LocalDateTime ,“本地...”部分表示没有特定的位置,因此没有时区,因此不是时间轴上的一个点——没有真正的意义。

    请记住,仅日期值的定义是不明确的。在任何给定时刻,世界各地的日期都不同。例如,在 Paris France 午夜刚过是新的一天,但在 Montréal Québec日期仍然是“昨天”。

    通常在业务中,某些时区是隐含的,甚至是无意识的直觉。从长远来看,关于数据点的无意识直觉往往效果不佳,尤其是在软件中。最好明确说明预期的时区。您可以将预期区域与日期一起存储,例如数据库表中的另一列,或者您可以在您的编程代码中进行注释。我相信存储日期时间值会更好、更安全。那么我们如何将仅日期转换为日期时间呢?

    新的一天通常是午夜过后的那一刻,也就是一天中的第一个时刻。您可能认为这意味着一天中的时间 00:00:00.0但不总是。 Daylight Saving Time (DST)并且可能其他异常情况可能会将第一时刻推向不同的 wall-clock time .让 java.time 确定通过 LocalDate 的第一时刻的正确时间。类及其 atStartOfDay 方法。
    ZoneId zoneId = ZoneId.of( "America/Montreal" );
    LocalDate today = LocalDate.now( zoneId );
    ZonedDateTime todayStart = today.atStartOfDay( zoneId );
    

    在某些业务环境中,新的一天可能被定义(或假定)为营业时间。例如,假设 New York 中的出版商当他们说“书稿将于 1 月 2 日到期”时,指的是本地时间上午 9 点。让我们获取该时区中该日期的时间。
    ZoneId zoneId = ZoneId.of( "America/New_York" );
    ZonedDateTime zdt = ZonedDateTime.of( 2016 , 1 , 2 , 9 , 0 , 0 , 0 , zoneId );
    

    这对在 New Zealand 工作的作者意味着什么? ?适应她particular time zone调用 withZoneSameInstant 向她介绍.
    ZoneId zoneId_Pacific_Auckland = ZoneId.of( "Pacific/Auckland" );
    ZonedDateTime zdt_Pacific_Auckland = zdt.withZoneSameInstant( zoneId_Pacific_Auckland );
    

    数据库

    对于数据库存储,我们转换为 Instant (UTC 时间线上的一个时刻)并作为 java.sql.Timestamp 传递如上所述。
    java.sql.Timestamp ts = java.sql.Timestamp.from( zdt.toInstant() );
    

    从数据库中检索时,转换回纽约日期时间。转换自 java.sql.TimestampInstant ,然后应用时区 ZoneId获得 ZonedDateTime .
    Instant instant = ts.toInstant();
    ZoneId zoneId = ZoneId.of( "America/New_York" );
    ZonedDateTime zdt = ZonedDateTime.ofInstant( instant , zoneId );
    

    如果您的 database driver符合JDBC 4.2或稍后,您可能能够传递/获取 java.time直接输入而不是转换为/从 java.sql类型。试试 PreparedStatement::setObject ResultSet::getObject 方法。

    关于java - 在 Web 应用程序中处理时区,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33343893/

    有关java - 在 Web 应用程序中处理时区的更多相关文章

    1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

      我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

    2. ruby - 将差异补丁应用于字符串/文件 - 2

      对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

    3. 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

    4. ruby - 在 Ruby 中编写命令行实用程序 - 2

      我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

    5. ruby-on-rails - Rails 应用程序之间的通信 - 2

      我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

    6. ruby - 无法运行 Rails 2.x 应用程序 - 2

      我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

    7. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

      刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

    8. java - 等价于 Java 中的 Ruby Hash - 2

      我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

    9. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

      我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

    10. java - 从 JRuby 调用 Java 类的问题 - 2

      我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

    随机推荐