There are many ways to format a string in Java. You can use the String.format() method, the printf() method, or the MessageFormat class for formatting strings.

Formatting a string using the String.format() method

The most common way of formatting a string in Java is using the String.format() method. This static method returns a formatted string using the given locale, format string, and arguments.

Here is an example:

String name = "John Doe";
String str = String.format("My name is %s", name);
System.out.println(str);
// My name is John Doe

Width and Padding

The String.format() method also allows us to set the width, alignment, and padding of the formatted string:

String message = "Hey 👋";

// Text width
String.format("|%20s|", message);
// |              Hey 👋|

// Left justify text
String.format("|%-20s|", message);
// |Hey 👋              |

// Maximum number of characters
String.format("|%.3s|", message);
// |Hey|

// Max. characters with width
String.format("|%20.3s|", message);
// |                 Hey|

Conversions

Besides formatting, the String.format() method can also be used to convert and format other data types into a string:

String str1 = String.format("%d", 999);     // Integer value
String str2 = String.format("%f", 56.7);    // Float value
String str3 = String.format("%x", 678);     // Hexadecimal value
String str4 = String.format("%o", 99);     // Octal value
String str5 = String.format("%tc", new Date());     // Date object
String str6 = String.format("%c", 'A');     // Char value

System.out.println(str1);
System.out.println(str2);
System.out.println(str3);
System.out.println(str4);
System.out.println(str5);
System.out.println(str6);

// 999
// 56.700000
// 2a6
// 143
// Sun Oct 03 14:05:44 PKT 2021
// A

Just like strings, you can also specify width and padding for other data types:

// Text width
String.format("|%10d|", 999);    // |       999|

// Justify left
String.format("|%-10d|", 999);  // |999       |

// Pad with zeros
String.format("|%010d|", 999);  // |0000000999|

// Positive number
String.format("%+d", 999);      // +999

// Thousands separator
String.format("%,d", 9999999);  // 9,999,999

// Enclose -ve number with parenthesis
String.format("%o", 99);    // (99)

// Alternative Hex & Octal formatting
String.format("%#x", 99);    // 0x63
String.format("%#o", 99);    // 0143

Read this article to learn more about integer-to-string conversion in Java.

Argument Index

The argument index is an integer indicating the position of the argument in the argument list. The first argument is referenced by 1$, the second by 2$, and so on.

String product = "Pizza 🍕";
double price = 5.99;

String str = String.format("The price of %2$s is USD %1$.2f", price, product);

System.out.println(str);
// The price of Pizza 🍕 is USD 5.99

With the argument index, the order of the arguments does not matter. So you can pass the arguments in any order.

Another way to reference arguments by position is to use the < flag that causes the argument for the previous format specifier to be re-used:

String str = String.format("Today is %1$tB %<td, %<tY", LocalDate.now());

System.out.println(str);
// Today is October 03, 2021

You can also print multiple specifiers to the same argument:

String str = String.format("%2$d + %2$d = 4", 1, 2);

System.out.println(str);
// 2 + 2 = 4

If there are more arguments than format specifiers, they are ignored.

Localization

By default, the String.format() method uses the default locale by calling the Locale.getDefault() method.

To localize the formatted string, you have to pass a valid locale as the first argument, as shown below:

int number = 4566778;
String withoutLocale = String.format("%,d", number);
String withLocale = String.format(Locale.GERMAN, "%,d", number);

System.out.println(withoutLocale); // 4,566,778
System.out.println(withLocale);    // 4.566.778

Notice the additional , flag before d in the format string. This flag is used to include locale-specific grouping separators.

Format Specifiers

Finally, here is a list of all format specifiers supported by the String.format() method:

Specifier Data Type Output
%a Floating point (except BigDecimal) Hex output of floating-point number
%b Any type “true” if non-null, “false” if null
%c Character Unicode character
%d Integer (incl. byte, short, int, long, BigInt) Decimal Integer
%e Floating point Decimal number in scientific notation
%f Floating point Decimal number
%g Floating point Decimal number, possibly in scientific notation depending on the precision and value.
%h Any type Hex String of value from hashCode() method.
%n None Platform-specific line separator.
%o Integer (incl. byte, short, int, long, BigInt) Octal number
%s Any type String value
%t Date/Time (includes long, Calendar, Date, and TemporalAccessor) %t is the prefix for date and time conversions. More formatting flags are needed after this. See the date and time conversion above. The full list can be found here
%x Integer (incl. byte, short, int, long, BigInt) Hex string

Formatting a string using the System.out.printf() method

Another way of formatting strings in Java is using the printf() method. This method prints the contents on the standard output stream and does not return anything:

System.out.printf("I am %s!%n", "John Doe");
System.out.printf("Total price is %.2f USD%n", 99.3);
System.out.printf("Today date is %tD%n", LocalDate.now());

// I am John Doe!
// Total price is 99.30 USD
// Today date is 10/03/21

The printf() method uses the same underlying principles as the String.format() method. This is because both these methods internally use the Formatter class to format strings. Thus, everything said for String.format() also applies to the printf() method.

The only difference is the return type; printf() prints the formatted string to the standard output stream (mostly console), whereas String.format() returns the formatted string. This makes String.format() more versatile as you can use the returned value in more than one way.

Formatting a string using the String.formatted() method

The String.formatted() method was introduced in Java 15 for formatting an instance of the String class using the supplied arguments.

This method is equivalent to String.format() with default locale, except that it must be called on an already declared instance of String:

String str = "Date is %tD and time is %tT";
String formatted = str.formatted(LocalDate.now(), LocalTime.now());

System.out.println(formatted);
// Date is 10/03/21 and time is 16:13:57

Formatting a string using the Formatter class

The above three methods use the Formatter class under to hood to format strings. You can use the same specifiers to format a string using any of them. They only differ on return type and whether to create an instance or not.

To use the Formatter class directly, you need to instantiate a new instance by using its constructor:

Formatter formatter = new Formatter();
formatter.format("Date is %tD and time is %tT", LocalDate.now(), LocalTime.now());

System.out.println(formatter);
// Date is 10/03/21 and time is 16:25:07

The Formatter class also works with StringBuilder making it a more flexible option:

StringBuilder builder = new StringBuilder();
Formatter formatter = new Formatter(builder);
formatter.format("Date is %tD and time is %tT", LocalDate.now(), LocalTime.now());
System.out.println(builder);
// Date is 10/03/21 and time is 16:30:32

Formatting a string using the MessageFormat class

Finally, the last way of formatting strings in Java that does not use Formatter under the hood is the MessageFormat class.

The MessageFormat class provides methods to create concatenated messages in a language-neutral way. It means the formatted messages will be the same regardless of the programming language used to construct them. This makes it suitable to create messages that are displayed to the end-users.

The MessageFormat class takes in a set of objects, formats them, and then replaces the patterns at the appropriate places with the formatted strings.

Here is an example:

String name = "John Doe";
String buy = "a coffee for $3.49";

String str = MessageFormat.format("{0} bought {1} on {2, date, long} at {2, time}.",
        name, buy, new Date());

System.out.println(str);
// John Doe bought a coffee for $3.49 on 3 October 2021 at 9:30:47 pm.

As you can see above, we are using curly brackets ({}) called a pattern for each of the arguments instead of percentage specifiers (%).

Let us look at the third pattern — {2, date, long}. The number 2 refers to the index of the argument that should be inserted in its place. In our case, it is a Date object.

The second part, date, represents the format type of the argument. There are four top-level format types: number, date, time, and choice.

The last part, long, represents the format style used to make a more specified selection. The date format type has short, medium, long, and full as format styles.

The MessageFormat class also supports localization of the messages:

MessageFormat form = new MessageFormat("{0} price starts from {1, number, currency}", 
        Locale.US);

System.out.println(form.format(new Object[]{"Tesla", 39900}));
// Tesla price starts from $39,900.00

For a complete list of format types and format styles, visit the documentation.

✌️ Like this article? Follow me on Twitter and LinkedIn. You can also subscribe to RSS Feed.