Groovy - Penutupan

Penutupan adalah blok kode anonim singkat. Biasanya hanya mencakup beberapa baris kode. Suatu metode bahkan dapat mengambil blok kode sebagai parameter. Mereka bersifat anonim.

Berikut adalah contoh penutupan sederhana dan seperti apa tampilannya.

class Example {
   static void main(String[] args) {
      def clos = {println "Hello World"};
      clos.call();
   } 
}

Dalam contoh di atas, baris kode - {println "Hello World"} dikenal sebagai closure. Blok kode yang direferensikan oleh pengenal ini dapat dijalankan dengan pernyataan panggilan.

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

Hello World

Parameter formal dalam penutupan

Closures juga dapat berisi parameter formal untuk membuatnya lebih berguna seperti metode di Groovy.

class Example {
   static void main(String[] args) {
      def clos = {param->println "Hello ${param}"};
      clos.call("World");
   } 
}

Dalam contoh kode di atas, perhatikan penggunaan $ {param} yang menyebabkan closure mengambil parameter. Saat memanggil closure melalui pernyataan clos.call, kita sekarang memiliki opsi untuk meneruskan parameter ke closure.

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

Hello World

Ilustrasi berikutnya mengulangi contoh sebelumnya dan menghasilkan hasil yang sama, tetapi memperlihatkan bahwa parameter tunggal implisit yang disebut dapat digunakan. Ini dia kata kunci di Groovy.

class Example {
   static void main(String[] args) {
      def clos = {println "Hello ${it}"};
      clos.call("World");
   } 
}

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

Hello World

Penutupan dan Variabel

Secara lebih formal, closure bisa merujuk ke variabel pada saat closure didefinisikan. Berikut adalah contoh bagaimana hal ini dapat dicapai.

class Example {     
   static void main(String[] args) {
      def str1 = "Hello";
      def clos = {param -> println "${str1} ${param}"}
      clos.call("World");
		
      // We are now changing the value of the String str1 which is referenced in the closure
      str1 = "Welcome";
      clos.call("World");
   } 
}

Dalam contoh di atas, selain meneruskan parameter ke closure, kita juga mendefinisikan variabel yang disebut str1. Penutupan juga mengambil variabel bersama dengan parameternya.

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

Hello World 
Welcome World

Menggunakan Penutupan dalam Metode

Penutupan juga dapat digunakan sebagai parameter untuk metode. Di Groovy, banyak metode bawaan untuk tipe data seperti List dan koleksi memiliki closure sebagai tipe parameter.

Contoh berikut menunjukkan bagaimana closure dapat dikirim ke metode sebagai parameter.

class Example { 
   def static Display(clo) {
      // This time the $param parameter gets replaced by the string "Inner"         
      clo.call("Inner");
   } 
	
   static void main(String[] args) {
      def str1 = "Hello";
      def clos = { param -> println "${str1} ${param}" }
      clos.call("World");
		
      // We are now changing the value of the String str1 which is referenced in the closure
      str1 = "Welcome";
      clos.call("World");
		
      // Passing our closure to a method
      Example.Display(clos);
   } 
}

Dalam contoh di atas,

  • Kami mendefinisikan metode statis yang disebut Tampilan yang mengambil penutupan sebagai argumen.

  • Kami kemudian mendefinisikan penutupan dalam metode utama kami dan meneruskannya ke metode Tampilan kami sebagai parameter.

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

Hello World 
Welcome World 
Welcome Inner

Penutupan dalam Koleksi dan String

Beberapa metode List, Map, dan String menerima closure sebagai argumen. Mari kita lihat contoh bagaimana closure dapat digunakan dalam tipe data ini.

Menggunakan Penutupan dengan Daftar

Contoh berikut menunjukkan bagaimana closure dapat digunakan dengan List. Dalam contoh berikut, pertama-tama kita mendefinisikan daftar nilai sederhana. Jenis kumpulan daftar kemudian mendefinisikan fungsi yang dipanggil.each. Fungsi ini mengambil closure sebagai parameter dan menerapkan closure ke setiap elemen daftar.

class Example {
   static void main(String[] args) {
      def lst = [11, 12, 13, 14];
      lst.each {println it}
   } 
}

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

11 
12 
13 
14

Menggunakan Closures dengan Maps

Contoh berikut menunjukkan bagaimana closure dapat digunakan dengan Maps. Dalam contoh berikut, pertama kita mendefinisikan Peta sederhana dari item nilai kunci. Jenis kumpulan peta kemudian mendefinisikan fungsi yang disebut .each. Fungsi ini mengambil closure sebagai parameter dan menerapkan closure ke setiap pasangan nilai kunci peta.

class Example {
   static void main(String[] args) {
      def mp = ["TopicName" : "Maps", "TopicDescription" : "Methods in Maps"]             
      mp.each {println it}
      mp.each {println "${it.key} maps to: ${it.value}"}
   } 
}

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

TopicName = Maps 
TopicDescription = Methods in Maps 
TopicName maps to: Maps 
TopicDescription maps to: Methods in Maps

Seringkali, kami mungkin ingin mengulang di seluruh anggota koleksi dan menerapkan beberapa logika hanya jika elemen tersebut memenuhi beberapa kriteria. Ini mudah ditangani dengan pernyataan bersyarat di closure.

class Example {
   static void main(String[] args) {
      def lst = [1,2,3,4];
      lst.each {println it}
      println("The list will only display those numbers which are divisible by 2")
      lst.each{num -> if(num % 2 == 0) println num}
   } 
}

Contoh di atas menunjukkan ekspresi kondisional if (num% 2 == 0) yang digunakan dalam closure yang digunakan untuk memeriksa apakah setiap item dalam daftar habis dibagi 2.

Saat kita menjalankan program di atas, kita akan mendapatkan hasil sebagai berikut -

1 
2 
3 
4 
The list will only display those numbers which are divisible by 2.
2 
4

Metode yang digunakan dengan Closures

Penutupan itu sendiri menyediakan beberapa metode.

Sr.No. Metode & Deskripsi
1 Temukan()

Metode find menemukan nilai pertama dalam koleksi yang cocok dengan beberapa kriteria.

2 Temukan semua()

Ia menemukan semua nilai dalam objek penerima yang cocok dengan kondisi penutupan.

3 any () & every ()

Metode setiap iterasi melalui setiap elemen koleksi memeriksa apakah predikat Boolean valid untuk setidaknya satu elemen.

4 mengumpulkan()

Metode mengumpulkan iterasi melalui pengumpulan, mengubah setiap elemen menjadi nilai baru menggunakan closure sebagai transformator.