Java Lambda表达式参数

时间:2016-09-11 18:27:25

标签: lambda java-8

我尝试使用Java中的Lambda表达式来理解以下方法参考代码:

interface MyFunc<T> {

    boolean func(T v1, T v2);
}

class HighTemp {

    private int hTemp;

    HighTemp(int ht) {
        hTemp = ht;
    }

    boolean sameTemp(HighTemp ht2) {
        return hTemp == ht2.hTemp;
    }

    boolean lessThanTemp(HighTemp ht2) {
        return hTemp < ht2.hTemp;
    }
}

class InstanceMethWithObjectRefDemo {

    static <T> int counter(T[] vals, MyFunc<T> f, T v) {
        int count = 0;
        for (int i = 0; i < vals.length; i++) {
            if (f.func(vals[i], v)) {
                count++;
            }
        }
        return count;
    }

    public static void main(String args[]) {
        int count;

        HighTemp[] weekDayHighs = {new HighTemp(89), new HighTemp(82),
            new HighTemp(90), new HighTemp(89),
            new HighTemp(89), new HighTemp(91),
            new HighTemp(84), new HighTemp(83)};

        count = counter(weekDayHighs, HighTemp::sameTemp,
                new HighTemp(89));
        System.out.println(count + " days had a high of 89");

        HighTemp[] weekDayHighs2 = {new HighTemp(32), new HighTemp(12),
            new HighTemp(24), new HighTemp(19),
            new HighTemp(18), new HighTemp(12),
            new HighTemp(-1), new HighTemp(13)};

        count = counter(weekDayHighs2, HighTemp::sameTemp,
                new HighTemp(12));
        System.out.println(count + " days had a high of 12");

        count = counter(weekDayHighs, HighTemp::lessThanTemp,
                new HighTemp(89));
        System.out.println(count + " days had a high less than 89");

        count = counter(weekDayHighs2, HighTemp::lessThanTemp,
                new HighTemp(19));
        System.out.println(count + " days had a high of less than 19");
    }
}

我的问题是,为什么函数sameTemp只有一个参数?接口声明它必须有2个参数,所以对我来说没有意义。也许hTemp实例变量对两者都足够了?我从Java中获取了完整的参考书,作者在那里解释了这一点:

  

块引用

     

在程序中,请注意HighTemp有两个实例方法:sameTemp( )和   lessThanTemp( )。如果两个HighTemp对象包含相同的内容,则第一个返回true   温度。如果调用对象的温度小于,则第二个返回true   传递对象的那个。每个方法都有一个HighTemp类型的参数,每个方法都有一个布尔结果。因此,每个都与MyFunc功能接口兼容   因为调用对象类型可以映射到func( )的第一个参数和。{   参数映射到func( )的第二个参数。

谢谢!

1 个答案:

答案 0 :(得分:10)

许多人在进行面向对象编程时忽略了......这是如何真正实现的。假设:

class C {
  public int foo(int parm) { return parm; }

关键是:在对象上“调用”方法的想法是抽象。你会如何在汇编程序中实现它;这样CPU就可以实际执行该代码。你意识到:“呼唤”是不可能的。在那个级别,你只在内存中的某个地方调用函数。但是一个函数如何知道它自己的对象?!

导致很少人知道的事情 - 您可以将上述代码编写为:

class C {
  public int foo(C this, int i) {
        System.out.println("i: " + i);
        return i;
   }
}

为了安全起见,我还在日食中写了一点测试:

@Test
public void testIt() {
    assertThat(new C().foo(5), is(5));
}

在我的日食中运行良好,单位测试通过,并打印“i:5”。

所以,整个秘诀是:在对象上定义方法时,总会有一个(通常是不可见的)第一个参数“this”给予该方法!

因此:虽然你的方法看起来只有一个参数,但是“missing”参数是你调用方法的对象!

从这个意义上说:boolean func(T v1, T v2)要求输入类型为T的两个参数。这两个方法(如boolean sameTemp(HighTemp ht2) ...)都可以写成boolean sameTemp(HighTemp this, HighTemp ht2) ; et voila:T型的两个参数。