Post

Ruby strftime, short and long story

strftime for the impatients

Sometimes examples worth a thousand words, so here are some that could help you right away (example inspired from APIDoks) :

1
2
3
4
5
6
d = DateTime.new(2021,11,19,8,37,48,"-06:00")
# => Fri, 19 Nov 2021 08:37:48 -0600
d.strftime("Printed on %m/%d/%Y")   
# => "Printed on 11/19/2021"
d.strftime("at %I:%M%p")
# => "at 08:37AM"

And a lot more of raw examples (explanations in paragraphs below)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
%Y%m%d           => 20211119                  
%F               => 2021-11-19                
%Y-%m            => 2021-11                   
%Y               => 2021                      
%C               => 20                        
%Y%j             => 2021323                   
%Y-%j            => 2021-323                  
%GW%V%u          => 2021W471                  
%G-W%V-%u        => 2021-W47-1                
%GW%V            => 2021W47                   
%G-W%V           => 2021-W47                  
%H%M%S           => 083748                    
%T               => 08:37:48                  
%H%M             => 0837                      
%H:%M            => 08:37                     
%H               => 08                        
%H%M%S,%L        => 083748,000                
%T,%L            => 08:37:48,000              
%H%M%S.%L        => 083748.000                
%T.%L            => 08:37:48.000              
%H%M%S%z         => 083748-0600               
%T%:z            => 08:37:48-06:00            
%Y%m%dT%H%M%S%z  => 20211119T083748-0600      
%FT%T%:z         => 2021-11-19T08:37:48-06:00 
%Y%jT%H%M%S%z    => 2021323T083748-0600       
%Y-%jT%T%:z      => 2021-323T08:37:48-06:00   
%GW%V%uT%H%M%S%z => 2021W471T083748-0600      
%G-W%V-%uT%T%:z  => 2021-W47-1T08:37:48-06:00 
%Y%m%dT%H%M      => 20211119T0837             
%FT%R            => 2021-11-19T08:37          
%Y%jT%H%MZ       => 2021323T0837Z             
%Y-%jT%RZ        => 2021-323T08:37Z           
%GW%V%uT%H%M%z   => 2021W471T0837-0600        
%G-W%V-%uT%R%:z  => 2021-W47-1T08:37-06:00    

Long story about strftime

Programmers need to constantly convert between different date and time formats. People in different parts of the world prefer dates to be displayed in different ways as is also true for many different systems that require dates to be in a specific format to be accepted. Catering for different date formats can be quite a pain. But it will remain a pain no longer because after this article you will become a wizard with the powers to easily format date/time in Ruby.

Ruby formats Time objects in a specific way by default. But you may want something else that can adapt to your case. Great! Ruby has a method specifically for this job. The ‘strftime’ (string format time) method can get you any format you will ever possibly need. It is a highly flexible method with a lot of options for you to experiment with. You can get the time without the date, or a nicely formatted date with the year, day & name of the current month.

It works by passing a string with format specifiers. These specifiers, more formally called directives, will be replaced by a value as they instruct the method to customize the resulting human readable date. If you have ever used the printf method the idea is very similar to that. Here are a few examples:

1
2
3
4
time = Time.new  
time.strftime("%d/%m/%Y") # "05/12/2015"
time.strftime("%I:%M %p") # "11:04 PM"
time.strftime("%d of %B, %Y") # "21 of December, 2015"

Before diving into the depths of the strftime method we need you to take this quick refresher on the Time class in Ruby so that you are fully prepared to gain the most out of this lesson.

The beginning of Time:

Although the strftime method is relevant to all time-related classes including Date, Time, and DateTime, we can get by with only understanding the Time class. The Time class in Ruby represents dates and times stored as the number of seconds that have elapsed since January 1, 1970 00:00 UTC. This time is known as the ‘Unix epoch’ and can be considered the starting point of time for computers. It has been used to grab the current time, date, and year since that time and until 2038. A time instance in Ruby holds both date and time. The date component consists of the day, month and year while the time component consists of the hour, minutes, and seconds.

In Ruby, you can get the current day and time using Time.new or Time.now and assign it to a variable. You can then get the year, month, day, hour, minute, second, etc in the following manner:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
t = Time.now()
puts t.year()
# => 2022

puts t.month()
# => 1

puts t.day()
# => 7

puts t.hour()
# => 22

puts t.min()
# => 0

puts t.sec()
# => 46

So what does a Time instance in Ruby look like on its own.

1
2
3
t = Time.now()
puts t
# => 2022-01-08 13:35:53 +0000

As you can see this format might be less than desirable in some circumstances. Before overloading you with all the possible combinations of directives we can provide the strftime method, let us first go through the real-world scenario where we want to send the front-end of our application a nicely formatted date that represents the joining date for a customer. Let’s learn how to do that in the next section where we go through the thought process of applying different directives to strftime to reach our desired result.

strftime : application and thought process

As stated earlier we want to display the data represented by the time instance such as 2022-01-08 13:35:53 +0000, into Saturday, 08 Jan 2022. The latter seems to be a much more common format that you would find on a website. So how do we get it? Firstly we need the full name of the day of the week i.e ‘Sunday’. By checking the ‘Weekday’ section of the cheat sheet shown later in this lesson we identify that we need to use the specifier %A..

1
2
3
t = Time.new(2022,1,8,13,35,53)
puts t.strftime("%A")
# => Saturday

Pretty cool right? Okay, maybe it doesn’t look that cool just yet. Let’s apply the rest of the directives. Next, I want to get, 08, the day of month padded with a zero if it’s a single digit. I simply tack on another directive, %d, that allows me to do just that. Similarly, for Jan, the abbreviated month we have %b and finally for 2022, the full numerical year, we have %Y. Altogether it becomes:

1
2
3
t = Time.new(2022,1,8,13,35,53)
puts t.strftime("%A %d %b %Y")
# => Saturday 08 Jan 2022

The date is now properly formatted but the observant reader might have noticed that there is still something missing. Yes, the ‘comma’ is missing. Points for you if you got it. There isn’t a directive or specifier for a comma or any string for that matter, but not to worry. Any string you provide that is not listed in the docs as a directive, will simply be output to the resulting string. Applying this information to the example above we simply add a comma after our first directive like so:

1
2
puts t.strftime("%A, %d %b %Y")
# => Saturday, 08 Jan 2022

Perfect !

You are now ready to be bombarded with as many directives as your heart desires.

Deep dive

This section first shows the basics of the strftime method, then contains a basic cheat sheet that you will need to get familiar with over time, and finally drives the concept home with some cool examples.

The strftime method requires a directive with the following structure and rules:

Structure - %<flags><width><modifier><conversion>

Rules:

  1. A directive starts with a percent (%) character.

  2. A flag and the conversion specifiers tell the method how to display the time and date.

  3. The minimum field width specifies the minimum width. You can ignore this.

  4. The modifiers are “E” and “O”. Can also be ignored.

  5. Regular text that isn’t listed in the directives will pass through as it is in the output.

strftime cheat sheet

Flags

SyntaxDescription
-no padding on a numerical output
_pad with spaces
0pad with zeros
^Upcase the result string
:Use colons for %z

Date (Year, Month, Day)

SyntaxDescription
%YYear with century
%yyear % 100 (00..99)
%mMonth of the year, zero-padded (01..12)
%BThe full month name (‘January’)
%bThe abbreviated month name (‘Jan’)
%dDay of the month, zero-padded (01..31)
%jDay of the year (001..366)

Time (Hour, Minute, Second):

SyntaxDescription
%HHour of the day, 24-hour clock, zero-padded (00..23)
%kHour of the day, 24-hour clock, blank-padded ( 0..23)
%IHour of the day, 12-hour clock, zero-padded (01..12)
%lHour of the day, 12-hour clock, blank-padded ( 1..12)
%PMeridian indicator, lowercase (‘am’ or ‘pm’)
%pMeridian indicator, uppercase (‘AM’ or ‘PM’)
%MMinute of the hour (00..59)
%SSecond of the minute (00..60)
%zTime zone as an hour and minute offset from UTC (e.g. +0900)

Weekday:

SyntaxDescription
%AThe full weekday name (e.g. Sunday)
%aThe abbreviated weekday name (e.g. Sun)
%uDay of the week starting Monday (1..7)

Useful Combinations:

SyntaxDescription
%cDate and time (%a %b %e %T %Y)
%DDate (%m/%d/%y)
%FThe ISO 8601 date format (%Y-%m-%d)
%r12-hour time (%I:%M:%S %p)
%R24-hour time (%H:%M)

For a more comprehensive list of options please visit the strftime method in Ruby Docs.

Last words : random examples

Initialize time:

1
t = Time.new(2022,1,8,13,35,53)

Time for your appointment:

1
2
puts t.strftime("%B, %Y at %-I:%M %p")
# => January, 2022 at 12:13 AM

Alarm Clock:

1
2
puts t.strftime("%r")
# => 12:13:29 AM

Windows OS date:

1
2
puts t.strftime("%d/%m/%Y")
# => 09/01/2022

Fun Fact Date:

1
2
puts t.strftime("%-d %b is on a %A and is day number %j of the year %Y")
# => 8 Jan is on a Saturday and is day number 008 of the year 2022

That’s pretty much it. Now go out there and make some unique looking date formats!

This post is licensed under CC BY 4.0 by the author.