Android Button押下処理をラムダ式、メソッド参照にしてみる

java8 で導入された
ラムダ式(Lambda) と メソッド参照 を
Androidの Button押下処理 で使ってみる。

個人的にメソッド参照は使い慣れてない。
メソッド参照が出てくると
「なんでこれ渡せるんじゃ?」
と戸惑ってしまうレベル。。

java8 設定

java8 にしてなければ Project Structure を開いて
sourceCompatibility と targetCompatibility を 1.8 に設定する。

[File] > [Project Structure]

 

匿名クラスで実装

まずはおなじみの 匿名クラス を使って
OnClick 処理をしてみる。

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Button button = findViewById(R.id.id_button);

        // 匿名クラス
        button.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View view) {
                showView(view);
            }
        });
    }

    private void showView(View view) {
        Toast.makeText(this, "Toast行きます!", Toast.LENGTH_SHORT).show();
    }
}

ボタン押下でトーストが表示されるだけ。

 

ラムダ式で実装

setOnClickListener() の引数を
匿名クラス から ラムダ式 に置き換えてみる

button.setOnClickListener(view -> {
    showView(view);
});

だいぶ簡素になった。

今回は showView(view) を呼ぶだけの一行の処理なので
さらに簡略化できる。

button.setOnClickListener(view -> showView(view));

中カッコ「{}」が取っ払われて見やすくなった。

 

メソッド参照で実装

ラムダ式 から メソッド参照 に置き換えてみる

button.setOnClickListener(this::showView);

こんな呼び方もできる。。
javaっぽくなくて変な気分になるが、これもちゃんと動く。

 

メソッド参照の注意

メソッド参照で気をつけたいのは

引数の個数と型が、抽象メソッドと代入したいメソッドで同じ必要がある
という点である。

つまり、今回の場合は
showView メソッドが以下の作りになっているとメソッド参照にできない。

// 引数がない(引数が1つ必要)
private void showView()

// 引数の型が違う(Viewにする必要がある)
private void showView(String str)

 

staticメソッドの場合

代入するメソッドが staticメソッド で メソッド参照する場合

protected void onCreate(Bundle savedInstanceState) {
    // 処理省略
    
    // メソッド参照(static)
    button.setOnClickListener(MainActivity::showViewStatic);
}

private static void showViewStatic(View view) {
}

 

まとめ

レイアウトを含めたソースコード。

MainActivity.java

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Button button = findViewById(R.id.id_button);

        // 匿名クラス
        button.setOnClickListener(new View.OnClickListener() {

            @Override
            public void onClick(View view) {
                showView(view);
            }
        });

        // ラムダ式
        button.setOnClickListener(view -> {
            showView(view);
        });
        
        // ラムダ式(中カッコ省略)
        button.setOnClickListener(view -> showView(view));

        // メソッド参照
        button.setOnClickListener(this::showView);

        // メソッド参照(static)
        button.setOnClickListener(MainActivity::showViewStatic);
    }

    private void showView(View view) {
        Toast.makeText(this, "Toast行きます!", Toast.LENGTH_SHORT).show();
    }

    private static void showViewStatic(View view) {
        Log.d("MainActivity", "Logcatで行きます!");
    }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/id_button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="押すボタン"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</androidx.constraintlayout.widget.ConstraintLayout>

開発日記

Posted by konp