web-dev-qa-db-ja.com

Pythonでdatetime.timeにN秒を追加する標準的な方法は何ですか?

Pythonでdatetime.timeの値が与えられた場合、それに整数の秒数を追加する標準的な方法はありますか。たとえば、11:34:59 + 3 = 11:35:02のようになります。

これらの明白なアイデアはうまくいきません。

>>> datetime.time(11, 34, 59) + 3
TypeError: unsupported operand type(s) for +: 'datetime.time' and 'int'
>>> datetime.time(11, 34, 59) + datetime.timedelta(0, 3)
TypeError: unsupported operand type(s) for +: 'datetime.time' and 'datetime.timedelta'
>>> datetime.time(11, 34, 59) + datetime.time(0, 0, 3)
TypeError: unsupported operand type(s) for +: 'datetime.time' and 'datetime.time'

最後に、私はこのような関数を書きました:

def add_secs_to_time(timeval, secs_to_add):
    secs = timeval.hour * 3600 + timeval.minute * 60 + timeval.second
    secs += secs_to_add
    return datetime.time(secs // 3600, (secs % 3600) // 60, secs % 60)

私はこれを行うためのより簡単な方法が欠けていると思って仕方がない。

関連する

307
Paul Stephenson

datetimeで完全なtimedelta変数を使用することができ、ダミーの日付を指定してからtimeを使用して時間の値を取得することもできます。

例えば:

import datetime
a = datetime.datetime(100,1,1,11,34,59)
b = a + datetime.timedelta(0,3) # days, seconds, then other fields.
print a.time()
print b.time()

3秒間隔で2つの値になります。

11:34:59
11:35:02

もっと読みやすいものを選ぶこともできます

b = a + datetime.timedelta(seconds=3)

あなたがそんなに傾いているのなら。


これを実行できる関数の後であれば、以下のaddSecsの使い方を調べることができます。

import datetime

def addSecs(tm, secs):
    fulldate = datetime.datetime(100, 1, 1, tm.hour, tm.minute, tm.second)
    fulldate = fulldate + datetime.timedelta(seconds=secs)
    return fulldate.time()

a = datetime.datetime.now().time()
b = addSecs(a, 300)
print a
print b

これは出力します:

 09:11:55.775695
 09:16:55
435
paxdiablo

ここで他の人が述べているように、あなたは単に全体のdatetimeオブジェクトを使うことができます:

sometime = get_some_time() # the time to which you want to add 3 seconds
later = (datetime.combine(date.today(), sometime) + timedelta(seconds=3)).time()

しかし、なぜ完全なdatetimeオブジェクトが必要なのかを説明する価値があると思います。午後11時に2時間足したとしたらどうなるでしょうか。正しい動作は何ですか?例外です。午後11時59分より長い時間を過ごすことはできないのですか。折り返すべきでしょうか。

プログラマーが異なれば期待することも異なるので、どちらを選んだ場合でも多くの人が驚きます。さらに悪いことに、プログラマーは最初にそれをテストしたときにうまく動作するようなコードを書いて、それから後で予期しないことをすることによってそれを壊してしまうでしょう。これは非常に悪いことです。したがって、timedeltaオブジェクトをtimeオブジェクトに追加することはできません。

45
Eli Courtwright

ちょっとしたこと、秒のデフォルト値を上書きするための明快さを追加するかもしれない

>>> b = a + datetime.timedelta(seconds=3000)
>>> b
datetime.datetime(1, 1, 1, 12, 24, 59)
19
unmounted

終日フルタイムを使用することを提案してくれた@Pax Diablo、@ bvmou、および@Arachnidに感謝します。もし私が外部のソースからdatetime.timeオブジェクトを受け取らなければならないなら、これは代わりのadd_secs_to_time()関数のようです:

def add_secs_to_time(timeval, secs_to_add):
    dummy_date = datetime.date(1, 1, 1)
    full_datetime = datetime.datetime.combine(dummy_date, timeval)
    added_datetime = full_datetime + datetime.timedelta(seconds=secs_to_add)
    return added_datetime.time()

この冗長コードは、このワンライナーに圧縮できます。

(datetime.datetime.combine(datetime.date(1, 1, 1), timeval) + datetime.timedelta(seconds=secs_to_add)).time()

とにかく、コードを明確にするために、それを関数にまとめたいと思います。

11
Paul Stephenson

それがあなたのプロジェクトに別のファイル/依存関係を追加する価値があるならば、私は算術演算をする能力でdatetime.timeを拡張する小さな小さなクラスを書いたところです。あなたが真夜中を過ぎて行くとき、それはゼロの周りに回ります。現在、「今から24時間後には何時になるでしょう」には、夏時間、うるう秒、過去のタイムゾーンの変更など、多くのコーナーケースがあります。しかし、時にはあなたは本当に単純なケースを必要とします、そしてそれはこれがすることです。

あなたの例は書かれるでしょう:

>>> import datetime
>>> import nptime
>>> nptime.nptime(11, 34, 59) + datetime.timedelta(0, 3)
nptime(11, 35, 2)

nptimedatetime.timeを継承しているので、これらのメソッドのどれもが使えるはずです。

PyPiからnptime( "非ペディキュアタイム")またはGitHubで入手できます。 https://github.com/tgs/nptime

6
rescdsk

使用されている単位が不明なため、単にdatetimeに数値を追加することはできません。秒、時間、週...

日付と時刻を操作するためのtimedeltaクラスがあります。 datetimeからdatetimeを引いた値はtimedeltaを表し、datetimetimedeltadatetimeを表します。2つのdatetimeは追加できませんが、2つのtimedeltaは追加できません。

何秒追加してtimedeltaオブジェクトを作成し、それをdatetimeオブジェクトに追加します。

>>> from datetime import datetime, timedelta
>>> t = datetime.now() + timedelta(seconds=3000)
>>> print(t)
datetime.datetime(2018, 1, 17, 21, 47, 13, 90244)

C++にも同じ概念があります:std::chrono::duration

5
user2387567

完全を期すために、これを arrow (Pythonの場合は日付と時刻の方が良い)で実行する方法を示します。

sometime = arrow.now()
abitlater = sometime.shift(seconds=3)
3
Bart Van Loon

datetime.datetimedatetime.timedeltaに追加してみてください。時間部分だけが必要な場合は、結果のdatetime.datetimeオブジェクトに対してtime()メソッドを呼び出して取得できます。

1
Nick Johnson

昔の質問ですが、タイムゾーンを処理する関数をスローすると思いました。キーパーツは、datetime.timeオブジェクトのtzinfo属性を結合に渡し、次に、結果のダミー日時でtimetz()の代わりにtime()を使用します。この回答は、ここの他の回答に一部影響を受けています。

def add_timedelta_to_time(t, td):
    """Add a timedelta object to a time object using a dummy datetime.

    :param t: datetime.time object.
    :param td: datetime.timedelta object.

    :returns: datetime.time object, representing the result of t + td.

    NOTE: Using a gigantic td may result in an overflow. You've been
    warned.
    """
    # Create a dummy date object.
    dummy_date = date(year=100, month=1, day=1)

    # Combine the dummy date with the given time.
    dummy_datetime = datetime.combine(date=dummy_date, time=t, tzinfo=t.tzinfo)

    # Add the timedelta to the dummy datetime.
    new_datetime = dummy_datetime + td

    # Return the resulting time, including timezone information.
    return new_datetime.timetz()

そして、ここに非常に単純なテストケースクラスがあります(組み込みunittestを使用):

import unittest
from datetime import datetime, timezone, timedelta, time

class AddTimedeltaToTimeTestCase(unittest.TestCase):
    """Test add_timedelta_to_time."""

    def test_wraps(self):
        t = time(hour=23, minute=59)
        td = timedelta(minutes=2)
        t_expected = time(hour=0, minute=1)
        t_actual = add_timedelta_to_time(t=t, td=td)
        self.assertEqual(t_expected, t_actual)

    def test_tz(self):
        t = time(hour=4, minute=16, tzinfo=timezone.utc)
        td = timedelta(hours=10, minutes=4)
        t_expected = time(hour=14, minute=20, tzinfo=timezone.utc)
        t_actual = add_timedelta_to_time(t=t, td=td)
        self.assertEqual(t_expected, t_actual)


if __== '__main__':
    unittest.main()
0
blthayer