zoukankan      html  css  js  c++  java
  • lambada 表达式

    1.lambada的存在来由

    匿名类的一个问题是,如果匿名类的实现非常简单,例如只包含一个方法的接口,那么匿名类的语法可能看起来不实用且不清楚。在这些情况下,您通常会尝试将功能作为参数传递给另一个方法,例如当有人单击按钮时应采取的操作。Lambda表达式使您可以执行此操作,将功能视为方法参数,或将代码视为数据。

    单 interface 单method

    public interface Predicate<T> {

    /**
    * Evaluates this predicate on the given argument.
    *
    * @param t the input argument
    * @return {@code true} if the input argument matches the predicate,
    * otherwise {@code false}
    */
    boolean test(T t);

    2.lambada理想用例

    参考 java.util.function 下面的个别functionnal interfaces

    Consequently, the JDK defines several standard functional interfaces, which you can find in the package java.util.function.

    定义一个 Person 类

    public class Person {
    
        public enum Sex {
            MALE, FEMALE
        }
    
        String name;
        LocalDate birthday;
        Sex gender;
        String emailAddress;
    
        public int getAge() {
            // ...
        }
    
        public void printPerson() {
            // ...
        }
    }

    有一个List<Person> 我们要打印出指定条件的Person信息,用lamdaba表达式实现可以这么做

    方式一


    public static void listConditionPerson(List<Person> person, Predicate<Person> pre, Consumer<Person> conf) {
    for (Person p : person) {
    if (pre.test(p))
    conf.accept(p);
    }
    }

    调用方式为:  
    listConditionPerson(person, person1 -> person1.getAge() > 10 && person1.getAge() <= 30, Person::printPerson);

    方式二:使用泛型(generics)的模式改造一下方法,可以支持不同队形的相同功能

    public static <X, Y> void processElements(
        Iterable<X> source,
        Predicate<X> tester,
        Function <X, Y> mapper,
        Consumer<Y> block) {
        for (X p : source) {
            if (tester.test(p)) {
                Y data = mapper.apply(p);
                block.accept(data);
            }
        }
    }

     调用方式为

     processElements(
    person,
    p -> p.getGender() == Person.Sex.MALE
    && p.getAge() >= 18
    && p.getAge() <= 25,
    p -> p.getEmailAddress(),
    email -> System.out.println(email)
    );

    方式三:使用支持lambada的聚合函数(stream api)

    person.stream().filter(person1 -> person1.getAge() > 10).map(p ->
    p.getEmailAddress()).forEach(email -> {
    System.out.println(email);
    });

    并发版
    person.parallelStream().filter(person1 -> person1.getAge() > 10).map(p ->
    p.getEmailAddress()).forEach(email -> {
    System.out.println(email);
    });

    lambda 语法官方介绍

    bda expression looks a lot like a method declaration; you can consider lambda expressions as anonymous methods—methods without a name.

    The following example, Calculator, is an example of lambda expressions that take more than one formal parameter:

    public class Calculator {
      
        interface IntegerMath {
            int operation(int a, int b);   
        }
      
        public int operateBinary(int a, int b, IntegerMath op) {
            return op.operation(a, b);
        }
     
        public static void main(String... args) {
        
            Calculator myApp = new Calculator();
            IntegerMath addition = (a, b) -> a + b;
            IntegerMath subtraction = (a, b) -> a - b;
            System.out.println("40 + 2 = " +
                myApp.operateBinary(40, 2, addition));
            System.out.println("20 - 10 = " +
                myApp.operateBinary(20, 10, subtraction));    
        }
    }
    
    

    The method operateBinary performs a mathematical operation on two integer operands. The operation itself is specified by an instance of IntegerMath. The example defines two operations with lambda expressions, addition and subtraction. The example prints the following:

    40 + 2 = 42
    20 - 10 = 10

    Accessing Local Variables of the Enclosing Scope

    Like local and anonymous classes, lambda expressions can capture variables; they have the same access to local variables of the enclosing scope. However, unlike local and anonymous classes, lambda expressions do not have any shadowing issues (see Shadowing for more information). Lambda expressions are lexically scoped. This means that they do not inherit any names from a supertype or introduce a new level of scoping. Declarations in a lambda expression are interpreted just as they are in the enclosing environment. The following example, LambdaScopeTest, demonstrates this:

    import java.util.function.Consumer;
    
    public class LambdaScopeTest {
    
        public int x = 0;
    
        class FirstLevel {
    
            public int x = 1;
    
            void methodInFirstLevel(int x) {
                
                // The following statement causes the compiler to generate
                // the error "local variables referenced from a lambda expression
                // must be final or effectively final" in statement A:
                //
                // x = 99;
                
                Consumer<Integer> myConsumer = (y) -> 
                {
                    System.out.println("x = " + x); // Statement A
                    System.out.println("y = " + y);
                    System.out.println("this.x = " + this.x);
                    System.out.println("LambdaScopeTest.this.x = " +
                        LambdaScopeTest.this.x);
                };
    
                myConsumer.accept(x);
    
            }
        }
    
        public static void main(String... args) {
            LambdaScopeTest st = new LambdaScopeTest();
            LambdaScopeTest.FirstLevel fl = st.new FirstLevel();
            fl.methodInFirstLevel(23);
        }
    }
    

    This example generates the following output:

    x = 23
    y = 23
    this.x = 1
    LambdaScopeTest.this.x = 0

    If you substitute the parameter x in place of y in the declaration of the lambda expression myConsumer, then the compiler generates an error:

    Consumer<Integer> myConsumer = (x) -> {
        // ...
    }

    The compiler generates the error "variable x is already defined in method methodInFirstLevel(int)" because the lambda expression does not introduce a new level of scoping. Consequently, you can directly access fields, methods, and local variables of the enclosing scope. For example, the lambda expression directly accesses the parameter x of the method methodInFirstLevel. To access variables in the enclosing class, use the keyword this. In this example, this.x refers to the member variable FirstLevel.x.

    However, like local and anonymous classes, a lambda expression can only access local variables and parameters of the enclosing block that are final or effectively final. For example, suppose that you add the following assignment statement immediately after the methodInFirstLevel definition statement:

    void methodInFirstLevel(int x) {
        x = 99;
        // ...
    }

    Because of this assignment statement, the variable FirstLevel.x is not effectively final anymore. As a result, the Java compiler generates an error message similar to "local variables referenced from a lambda expression must be final or effectively final" where the lambda expression myConsumer tries to access the FirstLevel.x variable:

    System.out.println("x = " + x);

    Target Typing

    How do you determine the type of a lambda expression? Recall the lambda expression that selected members who are male and between the ages 18 and 25 years:

    p -> p.getGender() == Person.Sex.MALE
        && p.getAge() >= 18
        && p.getAge() <= 25

    This lambda expression was used in the following two methods:

    When the Java runtime invokes the method printPersons, it's expecting a data type of CheckPerson, so the lambda expression is of this type. However, when the Java runtime invokes the method printPersonsWithPredicate, it's expecting a data type of Predicate<Person>, so the lambda expression is of this type. The data type that these methods expect is called the target type. To determine the type of a lambda expression, the Java compiler uses the target type of the context or situation in which the lambda expression was found. It follows that you can only use lambda expressions in situations in which the Java compiler can determine a target type:

    • Variable declarations

    • Assignments

    • Return statements

    • Array initializers

    • Method or constructor arguments

    • Lambda expression bodies

    • Conditional expressions, ?:

    • Cast expressions

    Target Types and Method Arguments

    For method arguments, the Java compiler determines the target type with two other language features: overload resolution and type argument inference.

    Consider the following two functional interfaces ( java.lang.Runnable and java.util.concurrent.Callable<V>):

    public interface Runnable {
        void run();
    }
    
    public interface Callable<V> {
        V call();
    }

    The method Runnable.run does not return a value, whereas Callable<V>.call does.

    Suppose that you have overloaded the method invoke as follows (see Defining Methods for more information about overloading methods):

    void invoke(Runnable r) {
        r.run();
    }
    
    <T> T invoke(Callable<T> c) {
        return c.call();
    }

    Which method will be invoked in the following statement?

    String s = invoke(() -> "done");

    The method invoke(Callable<T>) will be invoked because that method returns a value; the method invoke(Runnable) does not. In this case, the type of the lambda expression () -> "done" is Callable<T>.



    博客新手,勿喷!
  • 相关阅读:
    6:python2、python3 的区别及小数据池
    web前端----html表单操作
    web前端----html基础
    mysql数据库----索引原理与慢查询优化
    MySQL数据库----流程控制
    MySQL数据库----IDE工具介绍及数据备份
    MySQL数据库----数据锁
    MySQL数据库----事务处理
    MySQL数据库----事务
    MySQL数据库----函数
  • 原文地址:https://www.cnblogs.com/semonshi/p/11447494.html
Copyright © 2011-2022 走看看