I was planning to create a Guava tutorial. But it seems like it'll be too large for a single post, so I opted on splitting it into several parts. The first part contains everything related to Strings. Four main classes are explained:
- CharMatcher (which can be considered as a light form of JDK's Pattern+Matcher with string manipulation capabilities)
- Joiner and MapJoiner (which are useful for joining iterables or arrays into string representations)
- Splitter (which is split() of JDK on steroids).
CharMatcher can be thought as a Pattern+Matcher of JDK in a more simple and practical form. It's not a full fledged replacement because you can't use regular expressions as you do on JDK.
01 | String string = "Scream 4" ; |
04 | CharMatcher matcher = CharMatcher.JAVA_LETTER_OR_DIGIT; |
10 | int count = matcher.countIn(string); |
11 | System.out.println( "Letter or digit count: " +count); |
20 | System.out.println(matcher.matchesAllOf( "scream" )); |
22 | System.out.println(matcher.matchesAllOf( "scream " )); |
25 | System.out.println(matcher.matchesNoneOf( "_?=)(" )); |
You can negate the matcher so it accepts the complementary character set. e.g. if our CharMatcher was accepting {a, b, c}, it'll accept any character except {a, b, c}.
01 | CharMatcher negatedMatcher = matcher.negate(); |
15 | System.out.println(negatedMatcher.matchesAllOf( "scream" )); |
17 | System.out.println(negatedMatcher.matchesAllOf( "scream " )); |
19 | System.out.println(negatedMatcher.matchesNoneOf( "_?=)(" )); |
removeFrom() and retainFrom() are really convenient methods. The first one removes the matching string while the second one extracts the matching string.
01 | String review = "Scream 4 is the #1 teen-slasher!" ; |
02 | CharMatcher whitespaceMatcher = CharMatcher.JAVA_WHITESPACE; |
03 | String result = whitespaceMatcher.removeFrom(review); |
06 | System.out.println( "The sentence without whitespaces: " +result); |
16 | String result2 = CharMatcher.DIGIT.retainFrom(review); |
17 | System.out.println( "Retained digits: " +result2); |
indexIn() returns the index of the first matching character.
5 | int indexOfDigit = CharMatcher.DIGIT.indexIn(review); |
6 | System.out.println( "index Of Digit: " +indexOfDigit); |
Although it's possible to use CharMatcher with predefined matcher setting you can as well build your own.
01 | CharMatcher onlyEvenNumbersMatcher = CharMatcher.anyOf( "2468" ); |
03 | CharMatcher noEvenNumbersMatcher = CharMatcher.noneOf( "2468" ); |
06 | CharMatcher largeAtoZ = CharMatcher.inRange( 'A' , 'Z' ); |
07 | CharMatcher aToZ = CharMatcher.inRange( 'a' , 'z' ).or(largeAtoZ); |
13 | "Even numbers matcher result: " |
14 | +onlyEvenNumbersMatcher.matchesAllOf( "1354" )); |
18 | "Even numbers matcher result: " |
19 | +onlyEvenNumbersMatcher.matchesAllOf( "starwars" )); |
23 | "Even numbers matcher result: " |
24 | +onlyEvenNumbersMatcher.matchesAllOf( "2466" )); |
28 | "No even numbers matcher result: " |
29 | +noEvenNumbersMatcher.matchesAllOf( "1354" )); |
33 | "No even numbers matcher result: " |
34 | +noEvenNumbersMatcher.matchesAllOf( "1337" )); |
38 | "No even numbers matcher result: " |
39 | +noEvenNumbersMatcher.matchesAllOf( "supermario" )); |
43 | "a to Z matcher result: " +aToZ.matchesAllOf( "sezin" )); |
45 | "a to Z matcher result: " +aToZ.matchesAllOf( "Sezin" )); |
47 | "a to Z matcher result: " +aToZ.matchesAllOf( "SeZiN" )); |
49 | "a to Z matcher result: " +aToZ.matchesAllOf( "SEZIN" )); |
55 | "a to Z matcher result: " +aToZ.matchesAllOf( "scream4" )); |
You can use trimFrom(), trimLeadingFrom() and trimTrailingFrom() for enhanced trimming capability. Next class is the Joiner class. You probably know splitting capabilities of JDK. It's a mystery why a string joining mechanism is not added to JDK. Guava's Joiner is here to help you in case you need one. Joiner basically takes an iterable or an array and joins all the elements inside as Strings. After that, you can directly add it to a StringBuilder, an Appendable (like PrintWriter, BufferedWriter ... etc), or obtain a String in the "element1 SEPARATOR element2...." form. We choose the separator with on() method of Joiner class. It's possible to use a CharMatcher, a Pattern or a String as separator.
03 | ArrayList<string> charList = Lists.newArrayList( "a" , "b" , "c" , "d" ); |
04 | StringBuilder buffer = new StringBuilder(); |
08 | buffer = Joiner.on( "|" ).appendTo(buffer, charList); |
10 | "Joined char list appended to buffer: " +buffer.toString()); |
15 | String joinedCharList = Joiner.on( ", " ).join(charList); |
17 | "Joined char list as String: " +joinedCharList); |
24 | System.out.println(charList); |
30 | String join4 = Joiner.on( " - " ).skipNulls().join(charList); |
31 | System.out.println(join4); |
36 | join4 = Joiner.on( " - " ). |
37 | useForNull( "defaultValue" ).join(charList); |
38 | System.out.println(join4); |
If you have predefined String values no need to create an array or an iterable for joining them. Notice that you can join an arbitrary number of objects with the method below. The method works with var-args.
2 | join( "first" , "second" , "third" , "fourth" , "rest" ); |
3 | System.out.println(join4); |
Notice that if neither skipNulls() nor useForNull(String)is used, the joining methods will throw NullPointerException if any given element is null.
Joiner is for iterables and arrays. Joiner.MapJoiner inner class is the map counterpart of Joiner. You can join the map content directly using Joiner.MapJoiner class. First you have to build a Joiner and assign it a separator(1) using on(). Then you can call withKeyValueSeparator() which takes the separator(2) between key value pairs This map joiner can be used to join a map for obtaining a string or this can be appended to an Appendable. The form of the result is "key1 SEPARATOR(1) value1 SEPARATOR(2) key2 SEPARATOR(1) value2 SEPARATOR(2)..." without the empty spaces.
01 | Map < String, Long > employeeToNumber = Maps.newHashMap(); |
05 | employeeToNumber.put( "obi wan" , 1L); |
06 | employeeToNumber.put( "bobba" , 2L); |
09 | Joiner.on( "|" ).withKeyValueSeparator( "->" ); |
12 | String join5 = mapJoiner.join(employeeToNumber); |
13 | System.out.println(join5); |
Google Guava library contains a cool Splitter class that harness more power than the JDK's split functionality.
01 | String text = "I have to test my string splitter, |
02 | for this purpose I'm writing this text, "; |
11 | String[] split = text.split( "," ); |
12 | System.out.println(Arrays.asList(split)); |
I'd want to remove the empty elements and then trim each element to remove the unnecessary empty spaces before and after them. I can do this in several steps with the old splitter. It's quite easy with Guava's Splitter.
04 | Iterable<string> split2 = Splitter.on( "," ).omitEmptyStrings() |
05 | .trimResults().split(text); |
06 | System.out.println(Lists.newArrayList(split2)); |
14 | Iterable<string> split3 = Splitter.fixedLength( 5 ).split(text); |
15 | System.out.println(Lists.newArrayList(split3)); |
Notice that trimming is applied before checking for an empty result, regardless of the order in which the trimResults() and omitEmptyStrings() methods were invoked.
Strings class contains a number of utility methods.Most of them are checking String objects'. emptyToNull() and nullToEmpty() are quite similar. emptyToNull (nullToEmpty) returns the given string if it is non-empty (non-null) else it returns an empty (null) string.
1 | String emptyToNull = Strings.emptyToNull( "test" ); |
2 | System.out.println(emptyToNull); |
5 | emptyToNull = Strings.emptyToNull( "" ); |
6 | System.out.println(emptyToNull); |
isNullOrEmpty() is quite practical. I don't remember how many times I had to write (string != null && !string.isEmpty())in my code.
02 | boolean nullOrEmpty = Strings.isNullOrEmpty(arg); |
04 | System.out.println( "Null or Empty?: " +nullOrEmpty); |
08 | nullOrEmpty = Strings.isNullOrEmpty(arg); |
09 | System.out.println( "Null or Empty?: " +nullOrEmpty); |
13 | nullOrEmpty = Strings.isNullOrEmpty(arg); |
15 | System.out.println( "Null or Empty?: " +nullOrEmpty); |
I'll show you repeat() which returns a string consisting of the given number of concatenated copies of the input string.
1 | String repeat = Strings.repeat( "beetlejuice" , 3 ); |
2 | System.out.println(repeat); |
padEnd() and padStart() are quite similar. The first one adds the given char at the end of the given string as many times as the given integer value allows. The second one adds to the start.
01 | String padEnd = Strings.padEnd( "star wars" , 15 , 'X' ); |
02 | String padStart = Strings.padStart( "star wars" , 15 , 'X' ); |
03 | System.out.println( "padEnd: " +padEnd); |
06 | System.out.println( "padStart: " +padStart); |
09 | System.out.println(padStart.length() == 15 ); |
This is all for string-related classes of Guava.
Hi there, love your articles, very nice.
ReplyDeleteKeep it up!
Thanks for a good tech read
ReplyDeleteThanks, very informative with many nuggets
ReplyDeleteI liked it. Also i think there's mistake.
ReplyDeleteMap employeeToNumber = Maps.newHashMap();
Doesn't seem to be valid in Java ?
Map < String, Long > employeeToNumber = Maps.newHashMap();
ReplyDeleteshould work with guava. I had a typo there I assume.
Bro guava ile ilgili bir şeyler ararken istemeden bloguna denk geldim. Yabancı bir arkadaş bu sayfaya link vermiş. Dünya küçük işte. Burada da karşılaşmak varmış. Güzel iş çıkarmışsın. ;)
ReplyDelete