1. 程式人生 > >java裡的::也可以引用非靜態方法

java裡的::也可以引用非靜態方法

在java 1.8裡如下引用靜態方法的程式碼是可以編譯通過的:

// public static String format(String format, Object... args)
BiFunction<String, Object[], String> format = String::format;
System.out.println(format.apply("%s-%d-%x", new Object[]{"str", 6, 15}));

如下引用非靜態方法的程式碼也是可以編譯通過的:

// public String toLowerCase()
Function<String, String> toUpperCase = String::toUpperCase;
// public boolean isEmpty()
Function<String, Boolean> isEmpty = String::isEmpty;

但是下面程式碼不可以:

// public String concat(String str)
Function<String, String> fn = String::concat;

::符號是可以用來引用非靜態的方法,但條件是所用的方法模版第一個引數可以是所引用方法的例項且後面的引數和返回型別符合引用的非靜態發方法就可以,在呼叫時會用第一個引數這個例項去呼叫應用的方法。比如下面的方法是可以的:

public class Test {
	public static void main(String[] args) {
		// public Calc add(Number x)
		BiFunction<Calc, Number, Calc> fn = Calc::add;
		Number r = fn.apply(new Calc(10), 3).sub(4).add(1).getResult();

		System.out.println(r);
		
	}
}

class Calc {
	private Number value;
	
	public Calc() {
		value = 0;
	}
	
	public Calc(Number s) {
		if(s != null) {
			value = s;
		} else {
			value = 0;
		}
	}
	
	public Calc add(Number x) {
		value = value.doubleValue() + x.doubleValue();
		return this;
	}
	
	public Calc sub(Number x) {
		value = value.doubleValue() - x.doubleValue();
		return this;
	}
	
	public Number getResult() {
		return value;
	}
}