Javaの8機能インタフェースとラムダ研究ノート(B)

パートは、ブログのJava 8のラムダの研究ノート()関連の主要なレコードの知識ラムダ式は、このブログは、に焦点を当てる関数インタフェース

インターフェース(シングル抽象メソッド)のSAMタイプ:フロントも、実際には、関数インタフェースまたはインタフェースが本質的であるが、それは特別なインタフェースである、と述べました。

インターフェイス関数は、次の3つのプロパティがあります。

1.インターフェースは抽象メソッドではありません、そして何より、また少ないです。

両方の方法は、すでに実装されているため2.あなたは、静的メソッドと、デフォルトのメソッドを持つことができます。

3.抽象メソッドは、Objectクラスのメソッドと同じ名前であってはなりません。

@FunctionalInterface
interface Fun{
	//只设置一个抽象方法 ------可以正常运行
	abstract int spend(int remaining,int money);
	
	//加入静态方法和默认方法 ----可以正常运行
	static void staticMethod() {
		System.out.println("There is a static method!");
	}
	default void defaultMethod() {
		System.out.println("There is a default method!");
	}
	
	//设置Object中可以重写的抽象方法 ----编译报错
//	abstract boolean equals(Object obj);
	
}

Javaの8の新機能は、よりよい関数型プログラミングをサポートするために、関数インタフェースを提供します。Javaはまた、アノテーションを提供@FunctionalInterface注釈インターフェイスをマークし、コンパイラが自動的に私たちは、フォーマット機能のインターフェイスにインターフェイスが準拠かどうかをチェックするのに役立ちます。

具体的な使用とラムダ式

関数インタフェースは、表現ラムダの使用をサポートするために使用されます。つまり、それに対応するラムダ式のパラメータカスタムの関数インタフェースは、抽象メソッドのパラメータデフォルトで、このデフォルト値は、抽象戻り、戻り値です。 

package com.wjb.lambda;
/**
 * 函数式接口:
 * 		1.只能有一个抽象方法
		2.可以有静态方法和默认方法,因为这两种方法都是已经实现的了
		3.这个抽象方法一定不能跟Object类中的方法同名。
 * @author Administrator
 *
 */
@FunctionalInterface
interface Fun{
	//只设置一个抽象方法 ------可以正常运行
	abstract int spend(int remaining,int money);
	
	//加入静态方法和默认方法 ----可以正常运行
	static void staticMethod() {
		System.out.println("There is a static method!");
	}
	default void defaultMethod() {
		System.out.println("There is a default method!");
	}
	
	//设置Object中可以重写的抽象方法 ----编译报错
//	abstract boolean equals(Object obj);
	
}
public class FunInterface{
	public static void main(String[] args) {
		Fun fun = (a,b) -> (a-b); //参数就是spend()方法的参数,返回值a-b就是spend()方法的返回值
		System.out.println(fun.spend(100, 5));
	}
}

 5つの共通関数インタフェース

提供されているJavaの関数インタフェースjava.util.functionのパッケージには、もっと重要なのは、5つのインタフェースを以下の通りです。私たちを提供jdk8これらのインタフェースは、それはまた、より便利である公式の抽象インタフェースに応じて使用する場合、達成された(のに役立ついくつかの抽象的行動を私たちを)パターンの一部を提供することであると思います。 

1.Consumer(消費パターン)

機能:名前が示すように、一般的なオブジェクトが渡され、それがどのような値を返さない消費することです。このインターフェースは、任意の実用化シナリオを持っていること?あなたは基礎となるコードのforeachの方法を見れば実際には、このメソッドに必要なパラメータは、消費者インターフェースの型パラメータでなければなりません。

ビューの消費者インタフェースのソースコードは、唯一の方法は抽象的であることを知ることができます

void accept(T t);

それを実行するには、次の書き込みの例:

/**	
     * consumer接口测试	
     */	
    @Test	
    public void test_Consumer() {	
        //1. 使用consumer接口实现方法	
        Consumer<String> consumer = new Consumer<String>() {	
	
            @Override	
            public void accept(String s) {	
                System.out.println(s);	
            }	
        };	
        Stream<String> stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");	
        stream.forEach(consumer);	
	
        System.out.println("********************");	
	
        //2.使用lambda表达式,forEach方法需要的就是一个Consumer接口	
        stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");	
        Consumer<String> consumer1 = (s) -> System.out.println(s);//lambda表达式返回的就是一个Consumer接口	
        stream.forEach(consumer1);	
        //更直接的方式	
        //stream.forEach((s) -> System.out.println(s));	
        System.out.println("********************");	
	
        //3.使用方法引用,方法引用也是一个consumer	
        stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");	
        Consumer consumer2 = System.out::println;	
        stream.forEach(consumer);	
        //更直接的方式	
        //stream.forEach(System.out::println);	
    }

コード分​​析ショー:

リスト1では、我々は直接消費者インターフェースを作成し、実装方法は受け入れると呼ばれ、私たちが見つけた  forEach とき、出力することができ、対応する値を渡した後、パラメータの消費者タイプの必要性を。2コードでは、我々は消費者としてラムダ式を使用します。慎重にラムダ式の戻り値は、消費者である、あなたは見つけるでしょう見て、そう、あなたは理由を理解することができるようになります  forEach 方法ではなく、引数としてlamdda式を使用することができます。

コード3は、我々が使用方法を参照の消費者としての側面を、またに渡される  forEach 方法。

もちろん、上記消費者インターフェースに加えて、以下では、消費者にこれらのインタフェースを使用することができます。上記と同様の方法を使用して。IntConsumerDoubleConsumerLongConsumerBiConsumer

上記の例を読んだ後、私たちは二つの点のように要約することができます。

①消費者はインターフェースであり、限りの実現などの  accept 方法、「消費者」出力として使用することができます。
forEach メソッドのパラメータは、限り消費者のタイプに合わせて引用されたラムダ式、メソッドの戻り値として、彼らは消費者のタイプとして使用することができなければならない  forEach メソッドのパラメータ、および値を出力します。

 

2.Supplier(生産モデル)

機能:消費者インターフェースのコントラスト、サプライヤーインタフェースは、1つの値のみを返す、パラメータを渡していません。ソースコードを見てみましょう

T get();

理解することができ、それが返された値がどのような呼び出し元----ラムダ式のメーカーによって決定されているもの、任意のパラメータを受け取り、値を返しません。実際には、それが意味する容器そのようなインタフェースが使用される他の方法に使用することができ、データを格納するのに使用することができます。

それ以下の具体的な実現:

**	
     * Supplier接口测试,supplier相当一个容器或者变量,可以存储值	
     */	
    @Test	
    public void test_Supplier() {	
        //① 使用Supplier接口实现方法,只有一个get方法,无参数,返回一个值	
        Supplier<Integer> supplier = new Supplier<Integer>() {	
            @Override	
            public Integer get() {	
                //返回一个随机值	
                return new Random().nextInt();	
            }	
        };	
	
        System.out.println(supplier.get());	
	
        System.out.println("********************");	
	
        //② 使用lambda表达式,	
        supplier = () -> new Random().nextInt();	
        System.out.println(supplier.get());	
        System.out.println("********************");	
	
        //③ 使用方法引用	
        Supplier<Double> supplier2 = Math::random;	
        System.out.println(supplier2.get());	
    }

我々は、器具をサプライヤオブジェクトを作成する  get ことがあるので、従って、このインタフェースを使用するたびに値を返し、インタフェースに格納され、パラメータを持たず、値を返すメソッド容器

 ここでは、サプライヤーの特定の使用シナリオを見てみましょう。

/**	
     * Supplier接口测试2,使用需要Supplier的接口方法	
     */	
    @Test	
    public void test_Supplier2() {	
        Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5);	
        //返回一个optional对象	
        Optional<Integer> first = stream.filter(i -> i > 4)	
                .findFirst();	
	
        //optional对象有需要Supplier接口的方法	
        //orElse,如果first中存在数,就返回这个数,如果不存在,就放回传入的数	
        System.out.println(first.orElse(1));	
        System.out.println(first.orElse(7));	
	
        System.out.println("********************");	
	
        Supplier<Integer> supplier = new Supplier<Integer>() {	
            @Override	
            public Integer get() {	
                //返回一个随机值	
                return new Random().nextInt();	
            }	
        };	
	
        //orElseGet,如果first中存在数,就返回这个数,如果不存在,就返回supplier返回的值	
        System.out.println(first.orElseGet(supplier));	
    }	

取得対象のオプションを使用し、必要orElseGet orElse方法と、対象におけるサプライヤーオプションのインターフェースがあります。 

  • orElse:着信番号を元に戻すために、リターンの数の最初のいくつか、そうでない場合がある場合

  • orElseGet:いくつかのリターンの数が最初、そうでない場合がある場合、それはサプライヤーによって返された値を返します。

上記サプライヤインターフェースに加えて、以下はまた、サプライヤインターフェースを使用することができます。
IntSupplier 、DoubleSupplier 、LongSupplier 、BooleanSupplier上記と同様の方法を使用して。

サプライヤーの概要

①サプライヤインターフェースは、データを収容するための容器として理解することができます。
②サプライヤーインターフェースが持つ  get メソッドを、値を返します。

 

3.Predicate(分析モード-述部

機能:引数と戻りブール値を渡します。実際には、これはインタフェースの同様の判定bool型です。

 

または例を見て

/**	
     * Predicate谓词测试,谓词其实就是一个判断的作用类似bool的作用	
     */	
    @Test	
    public void test_Predicate() {	
        //① 使用Predicate接口实现方法,只有一个test方法,传入一个参数,返回一个bool值	
        Predicate<Integer> predicate = new Predicate<Integer>() {	
            @Override	
            public boolean test(Integer integer) {	
                if(integer > 5){	
                    return true;	
                }	
                return false;	
            }	
        };	
	
        System.out.println(predicate.test(6));	
	
        System.out.println("********************");	
	
        //② 使用lambda表达式,	
        predicate = (t) -> t > 5;	
        System.out.println(predicate.test(1));	
        System.out.println("********************");	
	
    }

このコードは、作成  Predicate インタフェースオブジェクトを、前記実装クラス  test メソッド、必要パラメータを渡すと返す  bool 値をので、界面の効果がされて決定その後、値を渡し、テストメソッドを呼び出し、それはブール値を返します。また、ラムダ式を返す使用できる  Predicate インターフェイスをして、呼び出す  test 方法!

さて、私たちは、述語の特定の使用シナリオを見に例を見て:

/**	
     * Predicate谓词测试,Predicate作为接口使用	
     */	
    @Test	
    public void test_Predicate2() {	
        //① 将Predicate作为filter接口,Predicate起到一个判断的作用	
        Predicate<Integer> predicate = new Predicate<Integer>() {	
            @Override	
            public boolean test(Integer integer) {	
                if(integer > 5){	
                    return true;	
                }	
                return false;	
            }	
        };	
	
        Stream<Integer> stream = Stream.of(1, 23, 3, 4, 5, 56, 6, 6);	
        List<Integer> list = stream.filter(predicate).collect(Collectors.toList());	
        list.forEach(System.out::println);	
	
        System.out.println("********************");	
	
    }

このコードは、最初の述語オブジェクトを作成し、実施する  test かを決定するための試験方法を行う、方法:受信パラメータが5より大きい場合、それは、そうでない場合はfalseを返すtrueを返し、次に呼び出す   メソッドを、 メソッドに必要なパラメータは、述語のインタフェースであります、限り存在としてある5件のより大きいデータが出力されます。Streamfilterfilter

2つの点を要約したものです。

①述語は、述語タイプのインターフェースは、実際には、唯一の裁判官の役割を果たしています。
②述語によって達成  test 判定方法を作ります。

 

4.機能(ファンクションモード)

機能:引数と戻り引数を渡します。関数は、であるの役割変換出力データの他の形式に入力されたデータを。

最初のインスタンス:

/**	
     * Function测试,function的作用是转换,将一个值转为另外一个值	
     */	
    @Test	
    public void test_Function() {	
        //① 使用map方法,泛型的第一个参数是转换前的类型,第二个是转化后的类型	
        Function<String, Integer> function = new Function<String, Integer>() {	
            @Override	
            public Integer apply(String s) {	
                return s.length();//获取每个字符串的长度,并且返回	
            }	
        };	
	
        Stream<String> stream = Stream.of("aaa", "bbbbb", "ccccccv");	
        Stream<Integer> stream1 = stream.map(function);	
        stream1.forEach(System.out::println);	
	
        System.out.println("********************");	
	
    }

出力は次のようになります。357

上記のコードは作成され  Function 、インタフェースオブジェクトを実装し  apply 、入力パラメータと出力パラメータを有する方法。前記第一パラメータは第二のタイプは、変換された後、変換する前に、一般的なタイプです。

上記のコードでは、することにある各文字列の長さが戻り値として返される文字列の長さを得ます。

で  Function 、私がいることを言わなければならない重要なアプリケーションインタフェース  Stream クラス  map メソッド、map 渡す方法  Function ポートは、変換後の返却  Streamクラスを。

上記機能インターフェイスに加えて、以下は、そのような機能インターフェイスを使用してもよいです。
IntFunction、DoubleFunction、LongFunction、ToIntFunction、 ToDoubleFunction、DoubleToIntFunction 上記と同様の方法を用いて、等。

 要約すると:

①機能インタフェースは、機能インタフェースは、変換されたエフェクトデータです。
②ファンクションインタフェース  apply 変換を行うための方法。

 

5.BiXXXX(例えばBiFunction)

機能:戻り値を得るために、2つのパラメータを渡します。 

抽象適用する唯一の方法です。

 R apply(T t, U u);

例で見てみましょう:

public class FunctionTest {

    public static void main(String[] args) {


        BiFunction<Integer,Integer,Integer>  biFunction= (i1,i2) -> i1+i2;

        System.out.println(biFunction.apply(1,2));
    }
}

 BiFuntion最初の二つの入力パラメータタイプ整数であり、整数は、パラメータタイプの最終結果です。

 

 

リファレンス

ラムダ式と機能インターフェース

インタビューでは、あなたがJavaの消費者、サプライヤー、述語と機能8 DOを理解してハングアップ?

公開された61元の記事 ウォンの賞賛9 ビュー30000 +

おすすめ

転載: blog.csdn.net/qq_33204444/article/details/105024889