Skip to content

Instantly share code, notes, and snippets.

@ygunayer
Created May 11, 2015 18:55
Show Gist options
  • Save ygunayer/c8941775a36cc2c60ad4 to your computer and use it in GitHub Desktop.
Save ygunayer/c8941775a36cc2c60ad4 to your computer and use it in GitHub Desktop.
Java 8 Lambda Expression Examples
package com.ygunayer.javastreams;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Optional;
import java.util.function.BiFunction;
import java.util.stream.Collectors;
public class Lambdas {
@FunctionalInterface
public interface FooConsumer<T, U, S> {
void accept(T t, U u, S s);
}
public class Person {
private String name;
private List<String> addresses;
private int age;
public Person(String name, List<String> addresses, int age) {
this.name = name;
this.addresses = addresses;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public List<String> getAddresses() {
return addresses;
}
public void setAddresses(List<String> addresses) {
this.addresses = addresses;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return name + " (" + age + ")";
}
}
public static void main(String[] args) {
Lambdas app = new Lambdas();
app.someMethod();
app.someOtherMethod();
app.someLambdaMethodCaller();
app.SomeLambdaGeneratorCaller();
app.optionalExample();
app.mapExample();
app.flatMapExample();
app.filterExample();
app.skipExample();
app.limitExample();
app.distinctExample();
app.sortExample();
app.reduceExample();
}
public void someMethod() {
FooConsumer<String, String, String> foo = (a, b, c) -> {
System.out.println(a + ", " + b + " and " + c);
};
// outputs "One, Two and Three"
foo.accept("One", "Two", "Three");
// outputs "Six, Nine and Ten"
foo.accept("Six", "Nine", "Ten");
}
public void someOtherMethod() {
BiFunction<Integer, Integer, String> foo = (a, b) -> {
return "The product of " + a + " and " + b + " is " + (a * b);
};
String foo1 = foo.apply(5, 10);
String foo2 = foo.apply(3, 5);
// returns "The product of 5 and 10 is 50"
System.out.println(foo1);
// returns "The product of 3 and -5 is -15"
System.out.println(foo2);
}
public void someLambdaMethodExecutor(BiFunction<Integer, Integer, Integer> fn) {
System.out.println("Result of fn(1, 2) is " + fn.apply(1, 2));
}
public void someLambdaMethodCaller() {
BiFunction<Integer, Integer, Integer> add = (a, b) -> {
return a + b;
};
// this is also valid
BiFunction<Integer, Integer, Integer> multiply = (a, b) -> a * b;
// outputs "Result of fn(1, 2) is 3"
someLambdaMethodExecutor(add);
// outputs "Result of fn(1, 2) is 2"
someLambdaMethodExecutor(multiply);
}
public BiFunction<Integer, Integer, Integer> SomeLambdaGenerator(String which) {
if ("add".equals(which))
return (a, b) -> {
return a + b;
};
else
return (a, b) -> {
return a * b;
};
}
public void SomeLambdaGeneratorCaller() {
BiFunction<Integer, Integer, Integer> add = SomeLambdaGenerator("add");
BiFunction<Integer, Integer, Integer> multiply = SomeLambdaGenerator("multiply");
// outputs "1 + 2 = 3"
System.out.println("1 + 2 = " + add.apply(1, 2));
// outputs "1 * 2 = 2"
System.out.println("1 * 2 = " + multiply.apply(1, 2));
}
public void optionalExample() {
Optional<String> someString = Optional.ofNullable("I'm here!");
// the type can be inferred, so no need to specify it explicitly
Optional<String> someAbsentString = Optional.ofNullable(null);
// outputs "Some String: I'm Here!"
someString.ifPresent(str -> System.out.println("Some String: " + str));
// does nothing
someAbsentString.ifPresent(str -> System.out.println("Some Absent String: " + str));
// outputs "someAbsentString present? No"
System.out.println("someAbsentString present? " + (someAbsentString.isPresent() ? "Yes" : "No"));
// this throws an exception!
// someAbsentString.get();
}
public void mapExample() {
List<Integer> numbers = new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5));
List<Integer> mapped = numbers.stream().map(number -> number * 5).collect(Collectors.toList());
// outputs "[5, 10, 15, 20, 25]"
System.out.println(mapped);
}
public void flatMapExample() {
Person john = new Person("John", Arrays.asList("John's Home", "John's Office"), 26);
Person mary = new Person("Mary", Arrays.asList("Mary's Home"), 25);
List<Person> people = new ArrayList<Person>(Arrays.asList(john, mary));
// notice how the return type is incorrect
List<List<String>> incorrect = people.stream().map(person -> person.getAddresses()).collect(Collectors.toList());
// notice how the return of Person::getAddresses() is turned back into a stream
List<String> allAddresses = people.stream().flatMap(person -> person.getAddresses().stream()).collect(Collectors.toList());
// outputs "[[John's Home, John's Office], [Mary's Home]]" which is incorrect
System.out.println(incorrect);
// outputs "[John's Home, John's Office, Mary's Home]"
System.out.println(allAddresses);
}
public void filterExample() {
List<Integer> numbers = new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5));
List<Integer> odds = numbers.stream().filter(number -> number % 2 == 1).collect(Collectors.toList());
// outputs "[1, 3, 5]"
System.out.println(odds);
}
public void skipExample() {
List<Integer> numbers = new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5));
List<Integer> remainder = numbers.stream().skip(3).collect(Collectors.toList());
// outputs "[4, 5]"
System.out.println(remainder);
}
public void limitExample() {
List<Integer> numbers = new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5));
List<Integer> taken = numbers.stream().limit(3).collect(Collectors.toList());
// outputs "[1, 2, 3]"
System.out.println(taken);
}
public void distinctExample() {
List<Integer> numbers = new ArrayList<Integer>(Arrays.asList(1, 1, 2, 3, 5));
List<Integer> distinct = numbers.stream().distinct().collect(Collectors.toList());
// outputs "[1, 2, 3, 5]"
System.out.println(distinct);
}
public void sortExample() {
Person john = new Person("John", Arrays.asList("John's Home", "John's Office"), 26);
Person mary = new Person("Mary", Arrays.asList("Mary's Home"), 25);
Person sean = new Person("Sean", Arrays.asList("Sean's Home"), 33);
List<Person> people = new ArrayList<Person>(Arrays.asList(john, mary, sean));
// this would fail since our Person class does not extend Comparable
// List<Person> sorted = people.stream().sorted().collect(Collectors.toList());
// this sorts people to their ages in a descending order
// it works because we're explicitly specifying how one person relates to another in terms of order
// also, if you wanted them in ascending order instead, simply reverse p1 and p2's ages in the subtraction
List<Person> sorted = people.stream().sorted((p1, p2) -> p2.getAge() - p1.getAge()).collect(Collectors.toList());
// outputs "[Sean (33), John (26), Mary (25)]"
System.out.println(sorted);
}
public void reduceExample() {
List<Integer> numbers = new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5));
int sum = numbers.stream().reduce(50, (a, b) -> a + b);
// we can emulate collect(Collectors.toList()) using the reduce operation!
// this overload of the method reduce() accepts three parameters: initial value, accumulator and combiner
// the accumulator function accumulates an item from the stream into the current accumulation
// and the combiner function is used to combine two accumulations in case they run in parallel
// so it's safe to say that the combiner function is a fail-safe mechanism for concurrency cases
List<Integer> asList = numbers.stream().reduce(new ArrayList<Integer>(), (list, number) -> {
list.add(number);
return list;
}, (list1, list2) -> {
list1.addAll(list2);
return list1;
});
// outputs "65"
System.out.println(sum);
// outputs "[1, 2, 3, 4, 5]"
System.out.println(asList);
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment