Thinking about using our OCA 8 book to study for the Java Foundations Junior Associate exam? It covers most of the topics. See what other topics you need to learn and where to read about that. One of those topics is formatting a String using %s, %n and %d. Conveniently for you , I had written a draft of this material for our OCP book before finding out that topic is not on the OCP 8 exam. I asked our publisher if I could post the material online and they said yes. I’ve edited it down to cover the subset that is on the Java Foundations Junior Associate exam. While the style matches our book, keep in mind it has not gone through the normal editing process by our publisher.
Note that the exam only expects you to know %d (digit/integer), %s (string) and %n (new line). This post covers a bit about %f (decimal) for two reasons:
- To help you remember why %d is digit/integer and not a floating point number
- To see the power of printf!
Formatting Strings
Formatting a String is useful because they let us specify a lot of complex logic clearly and succinctly . Imagine you had to write code to print out exactly 3 decimal places and align the decimal places. That would be a lot of coding and logic. Or it could be done using formatting:
List<Double> values = Arrays.asList(123.456, -1.4, 2.0); for (Double d : values) System.out.printf("%7.3f%n", d);Don’t worry if this seems like magic. In a nutshell, this uses a width of 7 with 3 decimal places followed by a new line. It outputs:
123.456 -1.400 2.000You don’t have to be able to write code like this. It is just to show the power of formatting. Also, there are a lot more types and options in existence than you need to know for the exam. See the Formatter class in the Javadoc for the ones not on the exam. The following figure shows what is going on from a high level. Java is substituting the parameter into a part of the format string.Ok. From this point on, you do need to know the material for the exam.
The PrintStream class defines the methods format and printf. They do the exact same thing. Java wanted to name the method format. However, C++ has had this method called printf for decades. Java supports both names to make things easy for both C++ developers and new developers.
System.out and System.err are both PrintStream instances. Remember that System.out.println() can show up in questions on just about anything as can System.out.printf(). This means you won’t know specifically if you are being tested on printf or something else.
The signatures of these two methods are:
- public PrintWriter format(String fmt, Object… args)
- public PrintWriter printf(String fmt, Object… args)
Remember that the three dots mean varargs. You can pass as many arguments as you want here. Also, remember that format and printf are exactly the same so we will start to use them interchangeably.
The first parameter is a String with embedded format parameters. The remaining parameters are an ordered list of the parameters to be inserted:
int num = 3; String s = "Three: "; System.out.format("%s %d", s, num);Java sees a format string of a String to be inserted, followed by a space and then followed by a number to be inserted. The output is:
Three: 3There are a number of parts to a format string. You only need to know a simplified version which looks like:
%[arg_index$]conversion_char
Conversion Character
%d and %s are format conversion specifiers. These format specifiers indicate the type of the parameter. You are only required to know three of the format specifiers for the exam as listed in following table. You do have to memorize these. They are copied from C++ so some of the types might seem odd to you. We’ve tried to give a description that helps you remember it rather than the shortest one possible. If you are a C++ developer, you might know that %d actually stands for decimal rather than digit. This is harder to remember for a new developer because the whole point of %d is that there is not a decimal point.
Remember that %d is digit (integer) and not double.
Conversion Specifier | Description |
%d | Specifies a number with digits and no decimal point. (also known as an integer number) |
%n | New line (line break) |
%s | Specifies a String |
Argument Index
The remaining part of a format string is the argument index. This is the index number followed by a $.
Argument indexes start counting from 1 instead of 0.
These examples should be straightforward:
System.out.printf("%s, %s", "play", "time");    // play, time System.out.printf("%2$s, %1$s", "play", "time"); // time, play System.out.printf("%2$s, %2$s", "play", "time"); // time, timeThe first line can omit the argument index since we want to use them in the order provided. The second line shows how to reverse the arguments. The final line shows how to use the same argument multiple times.
The Formatter Class
The Formatter class is an interpreter for the format strings used in format/printf methods. The class is similar to the PrintWriter class except that Formatter defines only the format methods and not the printf methods. Since the Formatter class didn’t exist in C++, Java didn’t need to be compatible with it. You invoke format in the way you do PrintWriter: passing in a format specifier followed by a comma-separated list of arguments.
For example, the following statements create a Formatter object and format a string of primitive types. What do you think this outputs?
StringBuilder sb = new StringBuilder(); Formatter fmt = new Formatter(sb); int x = 123; fmt.format("x=%d", x); System.out.println(sb.toString());In this example, the Formatter writes its output to a StringBuilder object. The int x is formatted and the resulting StringBuilder looks like this:
x=123As you can see, working with Formatter is similar to working with PrintWriter.
Summary
printf() and format() use a format string. You need to know the simplified format of: %[arg_index$]conversion_char. The arg_index is optional. However, it must be specified in the correct order if present. The components you need to know are:
- arg_index is an index that starts counting with one
- conversion_character includes:
-
- %b for boolean
- %d for digit/decimal/integer (no decimal point)
- %n for new line
- %s for strings
Practice Questions
Question 1
What is the result of the following code?
System.out.format("%2$d is bigger than %2$d", 10, 5);A: 10 is bigger than 5
B: 5 is bigger than 10
C: 10 is bigger than 10
D: 5 is bigger than 5
E: The code does not compile
F: A runtime exception is thrownQuestion 2
Which of the following are true about the format string passed to printf? (Choose all that apply)
A: The argument index is a 0 based index.
B: The argument index is a 1 based index.
C: double goes with %d
D: int goes with %dQuestion 3
What is the output of this statement?
System.out.printf("%2$d %1$s", "entry", 123.456);A: 123 entry
B: 123.456 entry
C: entry 123
D: entry 123.456
E: The code does not compile
F: A runtime exception is thrownQuestion 4
What is the output of this statement?
System.out.format("$n, $s", 123.456);A: %n %s
B: $n $s
C: %n, %s
D: $n, $s
E: None of the above
F: A runtime exception is thrownQuestion 5
What is the output of this statement?
System.out.println("%d", 123);A: 0
B: 123
C: %d
D: The code does not compile
E: A runtime exception is thrown