In this article, we are going to look at how to make use of generics to further add flexibility to our methods which accept lambdas. We will take a quick look at using named lambdas and we will finally look at the way we can use the bulk data aggregate functions provided in JDK8 to simplify our code.
We will continue to use the TestObject and ObjectTest classes in the GIT repository at java-blog-samples to expound on the topic.
Example 9
Now we are going to introduce generics to processObjects() in order to extend what we can do with lambdas and also to further decrease the brittleness of our code. We added the following method to cover this example:227 228 229 230 231 232 233 234 | private static <T, U> void processObjects(Iterable<T> collection, Predicate<T> filter, Function<T, U> mapper, Consumer<U> voidFunction){ for (T t : collection){ if (filter.test(t)){ U u = mapper.apply(t); voidFunction.accept(u); } } } |
Now our processing of the object can take any type of collection in and perform a filter on it and each filtered object can be mapped to return a different type object and then consume that object with a void function.
We implement this in our code as:
109 110 111 112 113 114 115 | //Ex. 9 - Use of Generics To Add Further Flexibility println( "Ex. 9 - Use of Generics To Add Further Flexibility" , under); println(); println( "TestObject name where a < b" ); println(); processObjects(objList, t -> t.getA() < t.getB(), t -> t.getName(), name -> println( "TestOblject (" + name + ")" )); println(); |
Ex. 9 - Use of Generics To Add Further Flexibility ************************************************** TestObject name where a < b TestObject (Lefty) TestObject (Windsurfer) TestObject (Shortstop) TestObject (Viper) TestObject (Stingray) TestObject (Snoopy) TestObject (Archangel) TestObject (Rainmaker) TestObject (Boss Hog) TestObject (Terror) |
Example 10
So what we have done so far is to effectively handle void returning methods using lambdas. How often, though, do you find yourself in a situation where you need to have a sublist of your main data? Why not use a lambda filter and generics to return you a sublist of any type based on any filter?We start by creating a method to do just this:
236 237 238 239 240 241 242 243 244 | private static <T, U> List<U> processObjects(Iterable<T> collection, Predicate<T> filter, Function<T,U> mapper){ List<U> uList = new ArrayList<U>(); for (T t : collection){ if (filter.test(t)){ uList.add(mapper.apply(t)); } } return uList; } |
Then we implement it in our code:
117 118 119 120 121 122 123 124 | //Ex. 10 - Use of Generics to Process Objects and Return Result Collection println( "Ex. 10 - Use of Generics to Process Objects and Return Result Collection" , under); println(); println( "TestObject count of names matched where a < b" ); println(); List<String> results = processObjects(objList, t -> t.getA() < t.getB(), t -> t.getName()); println( "The number of results in List returned was " + results.size()); println(); |
Ex. 10 - Use of Generics to Process Objects and Return Result Collection ************************************************************************ TestObject count of names matched where a < b The number of results in List returned was 10 |
Example 11
Next lets look at the use of bulk data aggregate methods which can be used to simplify some of the processes we have shown. Our new method looks like this:246 247 248 | private static <T> void processGenericObjects(Collection<T> collection, Predicate<T> criteria, Consumer<T> voidFunction){ collection.stream().filter(criteria).forEach(voidFunction); } |