DateTime.ToString()

wassupy.com

shorthand format strings

dates

d 3/14/2026
D Saturday, March 14, 2026
m March 14
M March 14
y March 2026
Y March 2026

times

t 2:04 PM
T 2:04:46 PM

combos

f Saturday, March 14, 2026 2:04 PM
F Saturday, March 14, 2026 2:04:46 PM
g 3/14/2026 2:04 PM
G 3/14/2026 2:04:46 PM
o 2026-03-14T14:04:46.5445241
r Sat, 14 Mar 2026 14:04:46 GMT
s 2026-03-14T14:04:46
u 2026-03-14 14:04:46Z
U Saturday, March 14, 2026 2:04:46 PM

custom date bits

era

%g AD
gg AD

year

yyyyy 02026
yyyy 2026
yyyy 2026
yy 26
y March 2026

month

MMMM March
MMM Mar
MM 03
%M 3

day

dddd Saturday
ddd Sat
dd 14
%d 14

custom time bits

hour

HH 14
%H 14
hh 02
%h 2

minute

mm 04
%m 4

second

ss 46
%s 46

subsecond

%f 5
ff 54
fff 544
ffff 5442
fffff 54428
ffffff 544288
fffffff 5442881

miscellaneous bits

date separator

%/ /

time separator

%: :

AM/PM

%t P
tt PM

time zone1

%K
%z +0
zz +00
zzz +00:00

Pitfalls and traps

General advice

Dates, times, and time zones are tricky. In .NET, there are subtle differences that can be challenging to spot. My general advice is:

Other gotchas

TimeZoneInfo.ConvertTime(...) returns a DateTime with Kind = Unspecified. For example:

DateTime dt = DateTime.UtcNow;
// dt.ToString("o") -> 2026-03-14T18:04:46.5452146Z
// dt.Kind -> Utc

TimeZoneInfo tz = TimeZoneInfo.FindSystemTimeZoneById("America/Chicago");
var sf = TimeZoneInfo.ConvertTime(d, tz);
// sf.ToString("o") -> 2026-03-14T13:04:46.5452146
// sf.Kind -> Unspecified

Note that the Unspecified time lacks a time zone offset in the output. You might be tempted to set its Kind to Local, but this is usually not a good idea:

// don't do this
var sfLocal = DateTime.SpecifyKind(sf, DateTimeKind.Local);
sfLocal.ToString("o") -> 2026-03-14T13:04:46.5452146+00:00

The time is correct, but the time zone offset is not. Chicago is -05:00, not +00:00. "Local" in this context is the server's time zone (-00:00).

1That undesired time zone also shows up in the related format strings %K, %z, zz, and zzz above. Again, these show the time zone of the server, not the time zone the value was converted to. Luckily, DateTimeOffset handles this better:

var dto = DateTimeOffset.UtcNow;
var chicagoDto = TimeZoneInfo.ConvertTime(dto, tz);
// dto.ToString("zzz")        --> +00:00
// chicagoDto.ToString("zzz") --> -05:00

Other surprising things

The Ticks property is not agnostic of time zones/kind:

DateTime utc = DateTime.UtcNow;
DateTime local = utc.ToLocalTime(); // don't do this
DateTime unspecified = TimeZoneInfo.ConvertTime(utc, tz);

// utc.Ticks         --> 639091082865453191
// local.Ticks       --> 639091082865453191
// unspecified.Ticks --> 639090902865453191

I guess it makes sense that Ticks would vary here, but I personally would have guessed that it was always in UTC, sort of like Javascript's getTime.

And DateTimeOffset works in the same spirit by factoring in the offset:

// these are the same for DateTimeOffset, but not DateTime
// dto.Ticks        --> 639091082865453126
// chicagoDto.Ticks --> 639090902865453126
//
// and the difference is the same as the time zone offset ✔️:
// (chicagoDto.Ticks - dto.Ticks) / TimeSpan.TicksPerHour --> -5