:start_time
および:timezone
属性を持つデータベースにレコードfoo
があります。
:start_time
はUTCの時刻です。たとえば、2001-01-01 14:20:00
です。 :timezone
は文字列です。たとえば、America/New_York
です。
:start_time
の値を持つ新しいTimeオブジェクトを作成しますが、そのタイムゾーンは:timezone
で指定されます。 Railsは賢く、UTCからの時間をそのタイムゾーンと一致するように更新するので、:start_time
をロードしてから:timezone
に変換したくありません。
現在、
t = foo.start_time
=> 2000-01-01 14:20:00 UTC
t.zone
=> "UTC"
t.in_time_zone("America/New_York")
=> Sat, 01 Jan 2000 09:20:00 EST -05:00
代わりに、私は見たいです
=> Sat, 01 Jan 2000 14:20:00 EST -05:00
すなわち。私はやってみたいです:
t
=> 2000-01-01 14:20:00 UTC
t.zone = "America/New_York"
=> "America/New_York"
t
=> 2000-01-01 14:20:00 EST
あなたがの線に沿って何かをしたいように聞こえます
ActiveSupport::TimeZone.new('America/New_York').local_to_utc(t)
つまり、このローカル時間(ゾーンを使用)をutcに変換します。 Time.zone
が設定されている場合は、もちろん
Time.zone.local_to_utc(t)
これはtにアタッチされたタイムゾーンを使用しません-変換元のタイムゾーンに対してローカルであると仮定します。
ここで保護する必要があるエッジケースの1つは、DSTの移行です。指定したローカル時間が存在しないか、あいまいになる場合があります。
Railsを使用している場合、Eric Walshの答えに沿った別の方法があります。
def set_in_timezone(time, zone)
Time.use_zone(zone) { time.to_datetime.change(offset: Time.zone.now.strftime("%z")) }
end
変換後、時間に時間オフセットを追加する必要があります。
これを行う最も簡単な方法は次のとおりです。
t = Foo.start_time.in_time_zone("America/New_York")
t -= t.utc_offset
なぜあなたがこれをしたいのかはわかりませんが、おそらく実際に時間を使って作業するのが最善です。時間とタイムゾーンをシフトする必要がある理由についての背景が役立つと思います。
実際には、次のように、変換後にオフセットを減算する必要があると思います。
1.9.3p194 :042 > utc_time = Time.now.utc
=> 2013-05-29 16:37:36 UTC
1.9.3p194 :043 > local_time = utc_time.in_time_zone('America/New_York')
=> Wed, 29 May 2013 12:37:36 EDT -04:00
1.9.3p194 :044 > desired_time = local_time-local_time.utc_offset
=> Wed, 29 May 2013 16:37:36 EDT -04:00
この時間をどこで使用するかによって異なります。
あなたの時間が属性である場合
時間を属性として使用する場合、同じ date_time_attribute gem を使用できます。
class Task
include DateTimeAttribute
date_time_attribute :due_at
end
task = Task.new
task.due_at_time_zone = 'Moscow'
task.due_at # => Mon, 03 Feb 2013 22:00:00 MSK +04:00
task.due_at_time_zone = 'London'
task.due_at # => Mon, 03 Feb 2013 22:00:00 GMT +00:00
別の変数を設定する場合
同じ date_time_attribute gem を使用します。
my_date_time = DateTimeAttribute::Container.new(Time.zone.now)
my_date_time.date_time # => 2001-02-03 22:00:00 KRAT +0700
my_date_time.time_zone = 'Moscow'
my_date_time.date_time # => 2001-02-03 22:00:00 MSK +0400
def relative_time_in_time_zone(time, zone)
DateTime.parse(time.strftime("%d %b %Y %H:%M:%S #{time.in_time_zone(zone).formatted_offset}"))
end
仕事を解決するために私が思いついた素早い小さな機能。誰かがこれを行うより効率的な方法を持っている場合、それを投稿してください!
TimeZonesとの格闘にもかなりの時間を費やし、Ruby 1.9.3をいじくり回した後、変換する前に名前付きタイムゾーンシンボルに変換する必要がないことに気付きました。
my_time = Time.now
west_coast_time = my_time.in_time_zone(-8) # Pacific Standard Time
east_coast_time = my_time.in_time_zone(-5) # Eastern Standard Time
これが意味するのは、あなたが望む地域で最初に適切な時間設定を得ることに集中し、あなたがそれについて考える方法(少なくとも私の頭ではこの方法でパーティション分割する)、そして最後にゾーンに変換できることですあなたのビジネスロジックを検証したい。
これはRuby 2.3.1でも機能します。
Ruby/Rails-値を変更せずにTimeのタイムゾーンを変更します)で投稿の元の著者によって要求されたものと同じことを行うヘルパーメソッドをいくつか作成しました。 。
また、私が観察したいくつかの特性を文書化しました。また、これらのヘルパーには、Railsフレームワークですぐに使用できない時間変換中に適用される自動夏時間を完全に無視するメソッドが含まれています:
def utc_offset_of_given_time(time, ignore_dst: false)
# Correcting the utc_offset below
utc_offset = time.utc_offset
if !!ignore_dst && time.dst?
utc_offset_ignoring_dst = utc_offset - 3600 # 3600 seconds = 1 hour
utc_offset = utc_offset_ignoring_dst
end
utc_offset
end
def utc_offset_of_given_time_ignoring_dst(time)
utc_offset_of_given_time(time, ignore_dst: true)
end
def change_offset_in_given_time_to_given_utc_offset(time, utc_offset)
formatted_utc_offset = ActiveSupport::TimeZone.seconds_to_utc_offset(utc_offset, false)
# change method accepts :offset option only on DateTime instances.
# and also offset option works only when given formatted utc_offset
# like -0500. If giving it number of seconds like -18000 it is not
# taken into account. This is not mentioned clearly in the documentation
# , though.
# Hence the conversion to DateTime instance first using to_datetime.
datetime_with_changed_offset = time.to_datetime.change(offset: formatted_utc_offset)
Time.parse(datetime_with_changed_offset.to_s)
end
def ignore_dst_in_given_time(time)
return time unless time.dst?
utc_offset = time.utc_offset
if utc_offset < 0
dst_ignored_time = time - 1.hour
elsif utc_offset > 0
dst_ignored_time = time + 1.hour
end
utc_offset_ignoring_dst = utc_offset_of_given_time_ignoring_dst(time)
dst_ignored_time_with_corrected_offset =
change_offset_in_given_time_to_given_utc_offset(dst_ignored_time, utc_offset_ignoring_dst)
# A special case for time in timezones observing DST and which are
# ahead of UTC. For e.g. Tehran city whose timezone is Iran Standard Time
# and which observes DST and which is UTC +03:30. But when DST is active
# it becomes UTC +04:30. Thus when a IRDT (Iran Daylight Saving Time)
# is given to this method say '05-04-2016 4:00pm' then this will convert
# it to '05-04-2016 5:00pm' and update its offset to +0330 which is incorrect.
# The updated UTC offset is correct but the hour should retain as 4.
if utc_offset > 0
dst_ignored_time_with_corrected_offset -= 1.hour
end
dst_ignored_time_with_corrected_offset
end
上記のメソッドをクラスまたはモジュールにラップした後、RailsコンソールまたはRubyスクリプトで試すことができる例:
dd1 = '05-04-2016 4:00pm'
dd2 = '07-11-2016 4:00pm'
utc_zone = ActiveSupport::TimeZone['UTC']
est_zone = ActiveSupport::TimeZone['Eastern Time (US & Canada)']
tehran_zone = ActiveSupport::TimeZone['Tehran']
utc_dd1 = utc_zone.parse(dd1)
est_dd1 = est_zone.parse(dd1)
tehran_dd1 = tehran_zone.parse(dd1)
utc_dd1.dst?
est_dd1.dst?
tehran_dd1.dst?
ignore_dst = true
utc_to_est_time = utc_dd1.in_time_zone(est_zone.name)
if utc_to_est_time.dst? && !!ignore_dst
utc_to_est_time = ignore_dst_in_given_time(utc_to_est_time)
end
puts utc_to_est_time
お役に立てれば。