preface

I used a lot of new things about Java8 in the project. At the beginning, I only knew how to write, but the writing was not very smooth. Then I found a good video about the new features of Java8 on the Internet, so I studied it. Here are my own notes and understandings of lambda expressions and the Stream API. Video address, interested can watch by themselves.

Java8 new features

  1. Faster replacement of data structures, memory structures (JVMS)
  2. * Less code (Lambda expressions) *
  3. * Powerful Stream API *
  4. Easy parallel fork join
  5. Minimize null pointer exceptions Optional

Lambda expressions

Before Lambda, let’s talk about functional interfaces. This is sort of the thing that exists for Lambda expressions. So what is a functional interface?

Functional interface

Definition: An interface has only one abstract interface. All interfaces under java.util. Function are functional interfaces. Java1.8 provides @functionalinterface to detect whether an interface is a FunctionalInterface. The Consumer interface code in the eg: java.util.function package

@FunctionalInterface
public interface Consumer<T> {

    void accept(T t);

JDK 1.8 interfaces can have default implementations
    default Consumer<T> andThen(Consumer<? super T> after) {
        Objects.requireNonNull(after);
        return(T t) -> { accept(t); after.accept(t); }; }}Copy the code

Lambda expressions are easy to understand once you know what a functional interface is.

“->” is the symbol of a lambda expression that represents the argument list of an abstract method in a functional interface on the left and your implementation of that method on the right. For example:

public class Test{
	public static void main(String[] args){
		 Consumer consumer = x-> System.out.println(x);
         consumer.accept(1); }}Copy the code

Output 1;

Four functional interfaces

When we use functional interfaces, we usually encapsulate them.

Consumer interface

The Consumer has only one abstract method named Accept, and the argument list has only a generic T with no return value. The data type of the parameter is determined by the class.

/ * * *@ClassName ConsumerTest
 * @DescriptionA consumptive interface that consumes string fields to print out *@Author ouyangkang
 * @DateThe 2019-02-18 him * * /
public class ConsumerTest {

    public static void main(String[] args) {
        test("hello",x-> System.out.println(x));
    }

    public static <T> void test(T t, Consumer<T> consumer) { consumer.accept(t); }}Copy the code

Output: Hello can also find functional interfaces such as ObjLongConsumer in the java.util.function package if multiple argument lists are required. Others can be checked

Supply interface

Supplier has only one abstract method named get with an empty argument list and a return value of type T.

/ * * *@ClassName SupplerTest
 * @DescriptionSupply interface string concatenation *@Author ouyangkang
 * @DateThe 2019-02-18 for * * /
public class SupplerTest {

    public static void main(String[] args) {
        String hello = test("hello ", () - >"word!");
        System.out.println(hello);
    }

    public static String test(String str,Supplier<String> supplier){
        returnstr + supplier.get(); }}Copy the code

The output is: Hello word! If you want to return data of a primitive type, you can find the corresponding functional interface under the java.util.function package, such as getAsLong

Functional interface

Function<T, R> has only one abstract method named apply, and the parameter list has only one parameter named T, which has a return value of data type R.

/ * * *@ClassName FunctionTest
 * @DescriptionThe functional interface converts the string to uppercase *@Author ouyangkang
 * @DateThe 2019-02-18 16:01 * * /
public class FunctionTest {


    public static void main(String[] args) {
        String test = test("hello", x -> x.toUpperCase());
        System.out.println(test);
    }


    public static String test(String str , Function<String,String> function){
        returnfunction.apply(str); }}Copy the code

If you need more than one input parameter and then return a value, you can find the corresponding functional interface such as BiFunction in the java.util.function package. Others can be checked

Assertion interface

The predicate type is also known as the judgment type. Predicate has only one abstract method named test, and the argument list has only one parameter T, which has a return value of type Boolean.

/ * * *@ClassName PredicateTest
 * @DescriptionPredicate interface to determine if the string size is greater than 6 *@Author ouyangkang
 * @DateThe company 2019-02-18 * * /
public class PredicateTest {
    public static void main(String[] args) {
        boolean hello = test("hello", x -> x.length() > 6);
        System.out.println(hello);
    }
    public static boolean test(String str, Predicate<String> predicate){
        returnpredicate.test(str); }}Copy the code

The output is: false

Stream API

One of the highlights of Java 8 is Stream, which is a completely different concept from InputStream and OutputStream in the Java.io package. Stream intermediate operations. Multiple intermediate operations can be connected to form a pipeline, and no processing will be performed in the middle unless termination is triggered on the pipeline! When the operation terminates, it is processed all at once, which is called lazy processing. To perform a stream operation, first fetch the stream. There are four methods to retrieve streams.

  1. Get streams through the Stream method provided by the collection family and parallelStream() (parallel streams) method
    public static void main(String[] args) {
        List<Integer> list = new ArrayList<>();
        // The common way to get streams
        Stream<Integer> stream = list.stream();
    }
Copy the code
  1. Arrays.stream() converts an array to a stream
    public static void main(String[] args) {
        int[] a = new int[] {1.2.3.4};
        IntStream stream = Arrays.stream(a);
        
    }
Copy the code
  1. Create a Stream with the stream.of today method
    public static void main(String[] args) {
        Stream<Integer> stream = Stream.of(1.2.3);
    }
Copy the code
  1. Creating an infinite stream
    public static void main(String[] args) {
        Stream<Integer> iterate = Stream.iterate(0, x -> x + 2);
    }
Copy the code

All convection operations can be divided into four types, namely, filtering and sharding, mapping, sorting, and termination (reduction and collection).

Filtering and sharding

Have the filter operation, distant and limit, the skip. Filter: indicates the filtering operation. The method parameter is assertion interface eg:

 public static void main(String[] args) {
        Stream<Integer> stream = Stream.of(1.2.3); stream.filter(x->x ! =2).forEach(x-> System.out.println(x));
    }
Copy the code

Output:

1
3
Copy the code

Distinct: indicates a deduplication operation. The method has no parameter limit: indicates that the first few data items are obtained and the method parameter is long skip: indicates that the first number of data items are skipped and the next ones are obtained. The method argument is long

mapping

Common operations include map and flatMap. Map: Processes the original data and returns the processed data. The method parameter is a functional interface. eg:

  public static void main(String[] args) {
       Stream<Integer> stream = Stream.of(1.2.3);
       stream.map(x->x*2).forEach(System.out::println);
    }
Copy the code

Output:

2
4
6
Copy the code

FlatMap: Consolidate the original data of the original stream into another stream one by one. Method takes a functional interface, but returns a stream. eg:

    public static void main(String[] args) {
        List<String> list = Arrays.asList("a"."b"."c");
        List<String> list2 = Arrays.asList("f"."d");
        list.stream().flatMap(x->list2.stream().map(y-> x + y)).forEach(System.out::println);
    }
Copy the code

The sorting

Sort (sort); sort (sort);

public static void main(String[] args) {
        Stream<Integer> stream = Stream.of(1.2.3);
        stream.sorted().forEach(System.out::println);
    }
Copy the code

Output:

1
2
3
Copy the code

Custom sorting

    public static void main(String[] args) {
        Stream<Integer> stream = Stream.of(1.2.3);
        stream.sorted((x,y)->-Integer.compare(x,y)).forEach(System.out::println);
    }
Copy the code

Output:

3
2
1
Copy the code

Termination of operations

  • AllMatch checks whether all elements are matched. The method parameter is an assertion interface
  • AnyMatch checks whether all elements are matched. The method parameter is an assertion interface
  • NoneMatch checks if all element method parameters are not matched as an assertion interface
  • FindFirst returns the first element with no method arguments
  • FindAny returns any element of the current stream with no method arguments
  • Count Returns the total number of elements in the stream
  • Max Indicates the maximum value of a return stream
  • Min Returns the minimum value in the stream with no method parameter

reduction

Reduce: Reduction — You can combine elements in a stream repeatedly to get a value. eg:

 public static void main(String[] args) {
        List<Integer> list1 = Arrays.asList(1.2.3.4.5.6.7.8.9.10);
        Integer reduce = list1.stream().reduce(11, (x, y) -> x + y);
        System.out.println(reduce);
    }
Copy the code

Output: 66

collect

This is a very common operation. Change the flow pack to another form. You receive an implementation of the Collector interface, a method for summarizing elements in the Stream. Collect using the collect method. Method parameter is Collector. Collectors can be implemented by static methods such as toList(), toSet(), and toMap(Function(T,R) key,Function(T,R) value).

  • ToList () returns a Collector that will input the element List to a new List.
  • toMap(Function
    keyMapper, Function
    valueMapper) returns a Collector that adds elements to a Map whose keys and values are the result of applying the provided mapping function to the input elements.
  • ToSet () returns a Collector that sets the input elements into a new Set. eg:

The User class

@Data
@ToString
public class User {
    private String name;

    private Integer age;

    private Integer salary;
}
Copy the code
   public static void main(String[] args) {
         List<User> users = Arrays.asList(new User("Zhang".19.1000),
                new User("Zhang".58.2000),
                new User("Bill".38.3000),
                new User("Zhao Wu".48.4000)); List<String> collect = users.stream().map(x -> x.getName()).collect(Collectors.toList()); Set<String> collect1 = users.stream().map(x -> x.getName()).collect(Collectors.toSet()); Map<Integer, String> collect2 = users.stream().collect(Collectors.toMap(x -> x.getAge(), x -> x.getName())); System.out.println(collect); System.out.println(collect1); System.out.println(collect2); }Copy the code

Output:

[zhang, zhang SAN, li si, Zhao Wu] [li si, zhang SAN, Zhao Wu] {48 = Zhao Wu, 19 = zhang SAN, 38 = li si, 58 = * *}Copy the code

grouping

The Collectors. GroupingBy () method is to return the Collector “implemented by the input element operation of the type on the group”, grouping elements according to the classification function. This is a very common operation. Let’s say you want to group people with the same name. groupingBy(Function<? super T,? extends K> classifier) eg:

    public static void main(String[] args) {
        List<User> users = Arrays.asList(new User("Zhang".19.1000),
                new User("Zhang".58.2000),
                new User("Bill".38.3000),
                new User("Zhao Wu".48.4000)); Map<String, List<User>> collect3 = users.stream().collect(Collectors.groupingBy(x -> x.getName())); System.out.println(collect3); }Copy the code

Output: {li si = [User {name = ‘bill’, the age = 38, salary = 3000}], threes = [User {name = ‘Joe’, the age = 19, salary = 1000}, User {name = ‘Joe’, the age = 58, Salary =2000}], salary= [User{name=’ salary ‘, age=48, salary=4000}]}

Of course, there are some other more complex grouping operations, the actual code to see the business to implement.

conclusion

Lambda expressions in java8 may not be very familiar at first, but once you are familiar with them, you will find that they are very useful, and lambda expressions can be combined with the stream API to write their own utility classes. In ordinary projects can be very time saving, improve the efficiency of writing code. I now present a List to Map utility class.

public class CollectionStream {
    public static void main(String[] args) {
        List<User> users = Arrays.asList(new User("Zhang".19.1000),
                new User("Zhang".58.2000),
                new User("Bill".38.3000),
                new User("Zhao Wu".48.4000)); Map<Integer, Integer> map = listToMap(users, x -> x.getAge(), x -> x.getSalary()); System.out.println(map); }/ * * *@Author ouyangkang
     * @DescriptionThe list keys to map keys must be different. If they are the same, an error will be reported. Method filters null for source data, key, and value. *@Date 9:27 2019/2/19
     * @paramSource Source data *@param key
     * @param value
     * @return java.util.Map<K,V>
    **/
    public static <DTO, K, V> Map<K, V> listToMap(List<DTO> source, Function<? super DTO, ? extends K> key, Function<? super DTO, ? extends V> value) {
        Objects.requireNonNull(source, "source not null");
        Objects.requireNonNull(key, "key not null");
        Objects.requireNonNull(value, "value not null"); Map<K, V> map = source.stream() .filter(dto -> dto ! =null) .filter(dto -> key.apply(dto) ! =null) .filter(dto -> value.apply(dto) ! =null)
                .collect(Collectors.toMap(key, value));
        returnmap; }}Copy the code