了解Lambda表达式

kar*_*ius 3 java lambda java-8

我不太清楚我是否理解Oracle Java教程中的Lambda表达式教程.令我困惑的主要问题是lambda的Index参数.ds.print(index ->{...} 编译器如何知道甚至是什么值索引?索引未在程序中的任何其他位置声明,因此索引参数甚至引用甚至是什么,编译器如何知道?

有问题:

public class DataStructure {

    private final static int SIZE = 15;
    private int[] arrayOfInts = new int[SIZE];

    public DataStructure() {
        for (int i = 0; i < SIZE; i++) {
            arrayOfInts[i] = i;
        }
    }

    public int size() {
        return SIZE;
    }

    public int get(int index) {
        return arrayOfInts[index];
    }

    interface DataStructureIterator extends java.util.Iterator<Integer> { }

    private class EvenIterator implements DataStructureIterator {

        private int nextIndex = 0;

        public boolean hasNext() {
            return (nextIndex <= SIZE - 1);
        }

        public Integer next() {
            Integer retValue = Integer.valueOf(arrayOfInts[nextIndex]);
            nextIndex += 2;
            return retValue;
        }
    }

    public DataStructureIterator getEvenIterator() {
        return new EvenIterator();
    }

    public void printEven() {
        DataStructureIterator iterator = getEvenIterator();
        while (iterator.hasNext()) {
            System.out.print(iterator.next() + " ");
        }
        System.out.println();
    }

    public void print(DataStructureIterator iterator) {
        while (iterator.hasNext()) {
            System.out.print(iterator.next() + " ");
        }
        System.out.println();
    }

    public void print(java.util.function.Function<Integer, Boolean> function) {
        for (int i = 0; i < SIZE; i++) {
            if (function.apply(i)) {
                System.out.print(arrayOfInts[i] + " ");
            }
        }
        System.out.println();
    }

    public static Boolean isEvenIndex(Integer index) {
        if (index % 2 == 0) return Boolean.TRUE;
        return Boolean.FALSE;
    }

    public static Boolean isOddIndex(Integer index) {
        if (index % 2 == 0) return Boolean.FALSE;
        return Boolean.TRUE;
    }

    public static void main(String s[]) {

        DataStructure ds = new DataStructure();

        System.out.println("printEven()");
        ds.printEven();

        System.out.println("print(DataStructureIterator) with "
                + "getEvenIterator");
        ds.print(ds.getEvenIterator());

        System.out.println("print(DataStructureIterator) with "
                + "anonymous class, odd indicies");
        ds.print(
                new DataStructure.DataStructureIterator() {
                    private int nextIndex = 1;
                    public boolean hasNext() {
                        return (nextIndex <= ds.size() - 1);
                    }
                    public Integer next() {
                        int retValue = ds.get(nextIndex);
                        nextIndex += 2;
                        return retValue;
                    }
                }
        );

        System.out.println("print(Function) with lambda expressions");
        ds.print(index -> {
            if (index % 2 == 0) return Boolean.TRUE;
            return Boolean.FALSE;
        });
        ds.print(index -> {
            if (index % 2 == 0) return Boolean.FALSE;
            return Boolean.TRUE;
        });

        System.out.println("print(Function) with method references");
        ds.print(DataStructure::isEvenIndex);
        ds.print(DataStructure::isOddIndex);
    }
}
Run Code Online (Sandbox Code Playgroud)

Boa*_*ann 6

ds.print方法采用类型的参数Function<Integer,Boolean>.所以这:

ds.print(index -> {
    if (index % 2 == 0) return Boolean.TRUE;
    return Boolean.FALSE;
});
Run Code Online (Sandbox Code Playgroud)

使用匿名类而不是lambda来表示与此语法等效:

ds.print(new Function<Integer,Boolean>() {
    @Override
    public Boolean apply(Integer index) {
        if (index % 2 == 0) return Boolean.TRUE;
        return Boolean.FALSE;
    }
});
Run Code Online (Sandbox Code Playgroud)

(Function该类的功能方法是apply.)

所以参数名index是任意的.你可以随意调用它.它只是lambda方法的本地名称.它的值由以下调用提供print:

if (function.apply(i)) ...
Run Code Online (Sandbox Code Playgroud)