秒数があります。 270921としましょう。xx日、yy時間、zz分、ww秒であると表示する方法はありますか?
divmod
を使用してかなり簡潔に行うことができます。
t = 270921
mm, ss = t.divmod(60) #=> [4515, 21]
hh, mm = mm.divmod(60) #=> [75, 15]
dd, hh = hh.divmod(24) #=> [3, 3]
puts "%d days, %d hours, %d minutes and %d seconds" % [dd, hh, mm, ss]
#=> 3 days, 3 hours, 15 minutes and 21 seconds
おそらく、DRY collect
、または多分inject
でクリエイティブを作成することでさらにそれを行うことができますが、コアロジックが3行の場合、やり過ぎになる可能性があります。
私はdivmodを使用するよりも簡単な方法があることを望んでいましたが、これが最もDRYであり、再利用可能な方法でした。
def seconds_to_units(seconds)
'%d days, %d hours, %d minutes, %d seconds' %
# the .reverse lets us put the larger units first for readability
[24,60,60].reverse.inject([seconds]) {|result, unitsize|
result[0,0] = result.shift.divmod(unitsize)
result
}
end
この方法は、フォーマット文字列と最初のインライン配列(つまり[24,60,60])を変更することで簡単に調整できます。
拡張バージョン
class TieredUnitFormatter
# if you set this, '%d' must appear as many times as there are units
attr_accessor :format_string
def initialize(unit_names=%w(days hours minutes seconds), conversion_factors=[24, 60, 60])
@unit_names = unit_names
@factors = conversion_factors
@format_string = unit_names.map {|name| "%d #{name}" }.join(', ')
# the .reverse helps us iterate more effectively
@reversed_factors = @factors.reverse
end
# e.g. seconds
def format(smallest_unit_amount)
parts = split(smallest_unit_amount)
@format_string % parts
end
def split(smallest_unit_amount)
# go from smallest to largest unit
@reversed_factors.inject([smallest_unit_amount]) {|result, unitsize|
# Remove the most significant item (left side), convert it, then
# add the 2-element array to the left side of the result.
result[0,0] = result.shift.divmod(unitsize)
result
}
end
end
例:
fmt = TieredUnitFormatter.new
fmt.format(270921) # => "3 days, 3 hours, 15 minutes, 21 seconds"
fmt = TieredUnitFormatter.new(%w(minutes seconds), [60])
fmt.format(5454) # => "90 minutes, 54 seconds"
fmt.format_string = '%d:%d'
fmt.format(5454) # => "90:54"
ご了承ください format_string
は、パーツの順序を変更させません(常に最も重要な値です)。より細かく制御するには、split
を使用して、値を自分で操作できます。
休憩が必要でした。これを打ち上げました:
s = 270921
dhms = [60,60,24].reduce([s]) { |m,o| m.unshift(m.shift.divmod(o)).flatten }
# => [3, 3, 15, 21]
Railsには、時間の距離を単語に変換するヘルパーがあります。その実装を見ることができます: distance_of_time_in_words
Railsを使用している場合、精度が必要ない場合は簡単な方法があります。
time_ago_in_words 270921.seconds.from_now
# => 3 days
この問題に対して私が見つけた最も簡単な方法を使用できます。
def formatted_duration total_seconds
hours = total_seconds / (60 * 60)
minutes = (total_seconds / 60) % 60
seconds = total_seconds % 60
"#{ hours } h #{ minutes } m #{ seconds } s"
end
必要に応じて、戻り値をいつでも調整できます。
2.2.2 :062 > formatted_duration 3661
=> "1 h 1 m 1 s"
Rubyを書き始めたところです。これは1.9.3専用です
def dateBeautify(t)
cute_date=Array.new
tables=[ ["day", 24*60*60], ["hour", 60*60], ["minute", 60], ["sec", 1] ]
tables.each do |unit, value|
o = t.divmod(value)
p_unit = o[0] > 1 ? unit.pluralize : unit
cute_date.Push("#{o[0]} #{unit}") unless o[0] == 0
t = o[1]
end
return cute_date.join(', ')
end
結果のサイズに基づいて動的な書式設定を追加するために、@ Mikeの回答を修正しました
def formatted_duration(total_seconds)
dhms = [60, 60, 24].reduce([total_seconds]) { |m,o| m.unshift(m.shift.divmod(o)).flatten }
return "%d days %d hours %d minutes %d seconds" % dhms unless dhms[0].zero?
return "%d hours %d minutes %d seconds" % dhms[1..3] unless dhms[1].zero?
return "%d minutes %d seconds" % dhms[2..3] unless dhms[2].zero?
"%d seconds" % dhms[3]
end