This is stated in a lot of places, but I think it worth mentioning always when we compare the timestamp with time zone
with timestamp without time zone
types: the timestamp WITH time zone
does not store the time zone information along with the timestamp. What it does is to store every data in UTC time zone, as stated in the docs:
For timestamp with time zone, the internally stored value is always in UTC (Universal Coordinated Time, traditionally known as Greenwich Mean Time, GMT). An input value that has an explicit time zone specified is converted to UTC using the appropriate offset for that time zone. If no time zone is stated in the input string, then it is assumed to be in the time zone indicated by the system's TimeZone parameter, and is converted to UTC using the offset for the timezone zone.
It is considered valid for some to use timestamp WITHOUT time zone
in situations where (1) everything is in the same timezone or (2) the application layer handles the time zone and just store everything in a certain time zone (usually UTC). But it is also considered an anti-pattern, simple because the correct solution for (1) is to configure the TimeZone setting to the given one timezone for the system and (2) is already solved, as PostgreSQL already stores everything on the same timezone (UTC).
Now, with those two down, I can came with only one good reason to use timestamp WITHOUT time zone
. That is when you want to store events in the future and that some kind of alert must be triggered when we got to that time. That could be good for timestamp WITH time zone
if, and only if, the rules defined by region's laws about time zone didn't ever change. The most common rule that changes is about the adoption or not of day light saving time (DST).
For example, imagine that you are at, let's say, 2013-06-15
(not yet in DST) schedule some event to happen at 2013-01-15 10:00
(which would be already in DST), and at 2013-06-15
your region was designated to adopt DST; but, some time after that, the government changed the rule and say your region will no longer use DST, and suddenly your scheduled time becomes 2013-01-15 11:00
(instead of 10:00
), that if you use timestamp WITH time zone
, and keep your TZ configurations up-to-date. Of course you may also notice that it is possible to treat such cases also with time zone, if you keep track of the rule changes in the regions/timezones of your interest, and update the affected records.
Worth mentioning that some regions does often change these rules (like at Brazil, some states - not the entire country - often change), but in most cases it changes it very earlier, so your users would be affected only by events scheduled very far from the current time.
With all that said, I only have one more suggestion. If you do have users, logs, or anything on different timezones, store the timezone they are coming from somewhere and choose and use timestamp with time zone
. That way you can (1) cross events happening closer to each other for different sources (independent of their timezones) and (2) show the original time (the clock time) the event has happened.
The WHERE
expression does not make sense:
WHERE date_trunc('day',NOW() - interval '1 day'
It has to evaluate to a boolean
value and you need to compare to an actual column of the table like (assuming a timestamp
, timestamptz
or date
column):
WHERE my_timestamp_column >= (NOW()::date - 1)
AND my_timestamp_column < NOW()::date
Postgres does not store a creation timestamp for table rows automatically. You need to add a column yourself to keep track (with default value and / or a trigger to make sure). Detailed instructions with code in this related answer:
"Commit timestamps" in Postgres 9.5 or later
You can activate the track_commit_timestamp
setting to start tracking. Then you can use pg_xact_commit_timestamp(xmin)
to retrieve the timestamp when each row was inserted or updated (after you started tracking). To get all rows that were inserted or updated within the last day:
SELECT * FROM mytab
WHERE pg_xact_commit_timestamp(xmin) > now() - interval '1 day'
Certain limitations apply. Commit timestamps are not kept indefinitely. Read the Postgres Wiki for details.
Once activated, commit timestamps are tracked for all transactions in the DB cluster, which causes a performance hit. If you only need it for one or a few tables, rather add timestamps to involved table(s) manually like instructed above.
Best Answer
to_timestamp()
expects the parameter to be given in seconds. Your value is in miliseconds. Just divide it by 1000:gives
2016-05-11 16:10:19.25+02