• How much do you know about new Java8 features?
  • How many new Java8 features have you used?
  • How well do you understand the new Java8 features?

All the new features of Java8

Lambda expressions, functional interfaces, interface default methods, interface static methods, extended annotations, repeated annotations, Optional, Stream, time/date apis, method references, parameter names preserved in bytecode, parallel arrays, CompletableFuture

Lambda expressions

Before JDK8, the parameters a method could take were variables, such as Object.method (object O). Like a callback. Then you might think of anonymous inner classes. For example, anonymous inner classes depend on interfaces, so you need to define an interface first

@FunctionalInterface
public interface PersonCallback {
    void callback(Person person);
}
Copy the code

The Person class:

public class Person {
    private int id;
    private String name;
    public Person(int id, String name) {
        this.id = id;
        this.name = name;
    }
    // After creating a Person, call back
    public static void create(Integer id, String name, PersonCallback personCallback) {
        Person person = newPerson(id, name); personCallback.callback(person); }}Copy the code
    public static void main(String[] args) {
        Person.create(1."Zhou yu".new PersonCallback() {
            public void callback(Person person) {
                System.out.println("Register..."); }}); Person.create(2."Teacher".new PersonCallback() {
            public void callback(Person person) {
                System.out.println("To land..."); }}); }Copy the code

The above PersonCallback is an action, but all we really care about is the contents of the callback method. Lambda represents the above code, which can be optimized to:

Person.create(1."Zhou yu", (Person person) -> {System.out.println("Register..."); })Copy the code

Do you notice how easy it is… The person. create method still receives the PersonCallback interface, but now passes a Lambda expression. Does the Lambda expression implement the interface? . Let’s look at the interface to Lambda expressions

Lambda allows functions to be arguments to a method. A Lambda is represented by a comma-separated argument list, a – > symbol, and a function body. For the expression above

  1. **(Person Person) is an entry to the Lambda expression, {system.out.println (” register…”) ); }** is the body of the function
  2. The important thing is that this expression has no name. We know that when we implement an interface, we must implement the methods in the interface. Now a Lambda expression should also follow this basic principle. What methods does a Lambda expression implement in the interface? The answer is: a Lambda expression implements the only and only one abstract method in an interface. So this kind of interface is called a functional interface. Lambda expressions actually fulfill the function of implementing interfaces and methods in the interface. We can also think of Lambda expressions as representing an action, and we can pass this particular action directly.

Of course, you can simplify the Lambda expression above:

Person.create(1."Zhou yu", person -> System.out.println("Register..."));
Copy the code

This is thanks to the type derivation mechanism of Java8. Since there is only one method in the interface, the Lambda expression must correspond to the implementation of the method. Since it is the only corresponding relation, the input parameter must be the Person class, so it can be shortened. And the method body has only one statement, so it can also be shortened to achieve the effect of concise expression.

Functional interface

Functional interfaces are a new interface definition. An interface decorated with ** @functionalinterface is called a FunctionalInterface **, or a FunctionalInterface is a normal interface with only one abstract method, and @functionalinterface can be used for validation. Only one abstract method of the following interface compiles correctly:

@FunctionalInterface
public interface TestFunctionalInterface {
    void test1(a);
}
Copy the code

The following interface has multiple abstract methods that compile errors:

@FunctionalInterface
public interface TestFunctionalInterface {
    void test1(a);
    void test2(a);
}
Copy the code

There are already some functional interfaces in JDK7, such as Runnable, Callable, FileFilter, etc. There are also many functional interfaces added in JDK8, such as the java.util.function package. For example, these four commonly used interfaces:

interface describe
Supplier Returns a result with no arguments
Function<T,R> Takes an input parameter and returns a result
Consumer Takes an input argument, returns no result
Predicate Takes an input parameter and returns a Boolean result.

So what’s the point of adding so many functional interfaces to Java8?

As we have analyzed above, a Lambda expression can also be interpreted as an implementer of a functional interface, but as an expression, it can be written in various ways, such as

  • () -> {return 0; }, no arguments passed in, return values
  • (int i) -> {return 0; }, pass in a parameter with a return value
  • (int I) -> {system.out.println (I)}, passes an argument of type int, but returns no value
  • (int I, int j) -> {system.out.println (I)}, pass in two arguments of type int, but return no value
  • (int i, int j) -> {return i+j; }, pass in two arguments of type int and return an int value
  • (int i, int j) -> {return i>j; }, pass in two arguments of type int, return a Boolean, and many, many more. Then thisEach expression should be written as an implementation class of a functional interface, requiring a specific functional interfaceFor example, the above four cases correspond to each otherSupplier<T>.Function<T,R>.Consumer<T>.BiConsumer<T, U>.BiFunction<T, U, R>.BiPredicate<T, U>.

The answer is obvious. Java8 provides so many functional interfaces to make it easier to write Lambda expressions. However, in special cases, you still need to define your own functional interfaces before writing Lambda expressions.

In general, Lambda expressions cannot be written without functional interfaces.

Default and static methods of the interface

In JDK7, if you want to add a method to an interface Collection, you need to change the source code of all its implementation classes (which is very scary). How was the design to solve this problem before Java8? There is now an interface PersonInterface with an abstract method in it:

public interface PersonInterface {
    void getName(a);
}
Copy the code

There are three implementation classes:

public class YellowPerson implements PersonInterface {

    @Override
    public void getName(a) {
        System.out.println("yellow"); }}public class WhitePerson implements PersonInterface {

    @Override
    public void getName(a) {
        System.out.println("white"); }}public class BlackPerson implements PersonInterface {

    @Override
    public void getName(a) {
        System.out.println("black"); }}Copy the code

Now I need to add a method to the PersonInterface interface, and all three of its implementation classes will need to be modified to compile, so I’m not going to do that here, so we can actually add an abstract class PersonAbstract at the beginning of the design, The three implementation classes inherit the abstract class. In this way, a new method is added to the PersonInterface interface. In fact, only the PersonAbstract class needs to be modified to implement the new method.

public interface PersonInterface {
    void getName(a);
    void walk(a);
}
public abstract class PersonAbstract implements PersonInterface {
    @Override
    public void walk(a) {
        System.out.println("walk"); }}public class BlackPerson extends PersonAbstract {
    @Override
    public void getName(a) {
        System.out.println("black"); }}public class WhitePerson extends PersonAbstract {
    @Override
    public void getName(a) {
        System.out.println("white"); }}public class YellowPerson extends PersonAbstract {
    @Override
    public void getName(a) {
        System.out.println("yellow"); }}Copy the code

In Java8, you can add implemented methods directly to the interface, either a Default method or a Static method.

The default method of the interface

Methods decorated with default on interfaces are called default methods. A default method in an interface must have a default implementation (method body) that the interface implementer can inherit or override.

    default void testDefault(a){
        System.out.println("default");
    };
Copy the code

Static methods of the interface

Methods decorated static in interfaces are called static methods.

    static void testStatic(a){
        System.out.println("static");
    };
Copy the code

Call method:

    TestInterface.testStatic();
Copy the code

Because of the default method and static method, you don’t have to change its implementation class, you can call it directly.

Repeated notes

Let’s say that we now have a service that we want to run periodically, just like Cron in Linux, and let’s say we want it to run every Wednesday at 12 o ‘clock, we might define an annotation that has two properties that represent the time.

public @interface Schedule {
    int dayOfWeek(a) default 1;   A few / / week
    int hour(a) default 0;    What time / /
}
Copy the code

So we can use this annotation on the corresponding service method to represent the running time:

public class ScheduleService {
    // Run every 3 days at 12:00
    @Schedule(dayOfWeek = 3, hour = 12)
    public void start(a) {
        // Execute the service}}Copy the code

So if we need this service to run every Thursday at 13:00, if it’s before JDK8, then… Embarrassed! You can’t compile errors like the code below

public class ScheduleService {
    // Two identical annotations in JDK will compile an error
    @Schedule(dayOfWeek = 3, hour = 12)
    @Schedule(dayOfWeek = 4, hour = 13)
    public void start(a) {
        // Execute the service}}Copy the code

In JDK8, you can change the code of the annotation to add @REPEATable meta annotation on the custom annotation and specify the store annotation of the duplicate annotation (in fact, you need an array to store the duplicate annotation).

@Repeatable(value = Schedule.Schedules.class)
public @interface Schedule {
    int dayOfWeek(a) default 1;
    int hour(a) default 0;

    @interfaceSchedules { Schedule[] value(); }}Copy the code

Meanwhile, the reflection related API provides a new function, getAnnotationsByType(), to return the type of the repeated annotation. Add the main method:

    public static void main(String[] args) {
        try {
            Method method = ScheduleService.class.getMethod("start");
            for (Annotation annotation : method.getAnnotations()) {
                System.out.println(annotation);
            }
            for (Schedule s : method.getAnnotationsByType(Schedule.class)) {
                System.out.println(s.dayOfWeek() + "|"+ s.hour()); }}catch(NoSuchMethodException e) { e.printStackTrace(); }}Copy the code

Output:

@repeatannotation.Schedule$Schedules(value=[@repeatannotation.Schedule(hour=12, dayOfWeek=3), @repeatannotation.Schedule(hour=13, dayOfWeek=4)])
3|12
4|13
Copy the code

Gets the name of the method parameter

Before Java8, it was very difficult to get Method parameter names using ASM, Javassist, and other techniques. Now, in Java8, you can get Method names directly from Method objects.

public class ParameterNames {
    public void test(String p1, String p2) {}public static void main(String[] args) throws Exception {
        Method method = ParameterNames.class.getMethod("test", String.class, String.class);

        for(Parameter parameter : method.getParameters()) { System.out.println(parameter.getName()); } System.out.println(method.getParameterCount()); }}Copy the code

Output:

arg0
arg1
2
Copy the code

The number of parameters is correct, but the names are incorrect. You need to add the ** -parameters ** parameter at compile time before running it. Add:

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <version>3.1</version>
    <configuration>
        <compilerArgument>-parameters</compilerArgument>
        <source>1.8</source>
        <target>1.8</target>
    </configuration>
</plugin>
Copy the code

The output becomes:

p1
p2
2
Copy the code

CompletableFuture

When we use Javer to talk about asynchronous calls, we naturally think of Future, as in:

public class FutureDemo {

    /** * perform a calculation asynchronously *@param args
     */
    public static void main(String[] args) {

        ExecutorService executor = Executors.newCachedThreadPool();
        Future<Integer> result = executor.submit(new Callable<Integer>() {
            public Integer call(a) throws Exception {
                int sum=0;
                System.out.println("Calculating...");
                for (int i=0; i<100; i++) {
                    sum = sum + i;
                }
                Thread.sleep(TimeUnit.SECONDS.toSeconds(3));
                System.out.println("That's it!);
                returnsum; }}); System.out.println("Do something else...");

        try {
            System.out.println("result:" + result.get());
        } catch (Exception e) {
            e.printStackTrace();
        }

        System.out.println("It's all done..."); executor.shutdown(); }}Copy the code

Now, what if I wanted to do this asynchronously and then I could take this result and do something else asynchronously? The problem is that one thread depends on another thread, and the Future is inconvenient. Let’s look at the implementation of CompletableFuture:

public static void main(String[] args) {

        ExecutorService executor = Executors.newFixedThreadPool(10);


        CompletableFuture result = CompletableFuture.supplyAsync(() -> {
            int sum=0;
            System.out.println("Calculating...");
            for (int i=0; i<100; i++) {
                sum = sum + i;
            }
            try {
                Thread.sleep(TimeUnit.SECONDS.toSeconds(3));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"That's it!);
            return sum;
        }, executor).thenApplyAsync(sum -> {
            System.out.println(Thread.currentThread().getName()+"Print"+sum);
            return sum;
        }, executor);


        System.out.println("Do something else...");

        try {
            System.out.println("result:" + result.get());
        } catch (Exception e) {
            e.printStackTrace();
        }

        System.out.println("It's all done...");

        executor.shutdown();
    }
Copy the code

Results:

Calculating... Do other things... Pool-1-thread-1 = pool-1-thread-1 = pool-1-thread-1 = pool-1-thread-1 = pool-1-thread-1 Pool-1-thread-2 print 4950 result:4950 Everything is done...Copy the code

Simply use thenApplyAsync to do this. Of course CompletableFuture has a lot of other features that we’ll cover in a separate section next time.

Java8 features include Stream and Optional. These two features are also widely used. I believe many students have already heard about them and have already understood them. There are a lot of new features, although currently Java11 has been released, the features in Java8 will definitely be retained in subsequent releases. As for these new features in this article, we estimate that we will use less, so we have this article to carry out a popularization, hope that they will be received.