Clojure - Penanganan Pengecualian

Exception handlingdiperlukan dalam bahasa pemrograman apa pun untuk menangani error runtime sehingga aliran normal aplikasi dapat dipertahankan. Pengecualian biasanya mengganggu aliran normal aplikasi, itulah alasan mengapa kita perlu menggunakan penanganan pengecualian dalam aplikasi kita.

Pengecualian secara luas diklasifikasikan ke dalam kategori berikut -

  • Checked Exception- Kelas yang memperluas kelas Throwable kecuali RuntimeException dan Error disebut sebagai pengecualian yang dicentang. Misalnya IOException, SQLException, dll. Pengecualian yang dicentang diperiksa pada waktu kompilasi.

Mari kita pertimbangkan program berikut yang melakukan operasi pada file bernama Example.txt. Namun, selalu ada kasus di mana file Example.txt tidak ada.

(ns clojure.examples.example
   (:gen-class))

;; This program displays Hello World
(defn Example []
   (def string1 (slurp "Example.txt"))
   (println string1))
(Example)

Jika file Example.txt tidak ada, pengecualian berikut akan dibuat oleh program.

Caused by: java.io.FileNotFoundException: Example.txt (No such file or
directory)
at java.io.FileInputStream.open0(Native Method)
at java.io.FileInputStream.open(FileInputStream.java:195)
at java.io.FileInputStream.<init>(FileInputStream.java:138)
at clojure.java.io$fn__9185.invoke(io.clj:229)
at clojure.java.io$fn__9098$G__9091__9105.invoke(io.clj:69)
at clojure.java.io$fn__9197.invoke(io.clj:258)
at clojure.java.io$fn__9098$G__9091__9105.invoke(io.clj:69)

Dari pengecualian di atas, kita dapat dengan jelas melihat bahwa program memunculkan FileNotFoundException.

  • Unchecked Exception- Kelas yang memperluas RuntimeException dikenal sebagai pengecualian yang tidak dicentang. Misalnya, ArithmeticException, NullPointerException, ArrayIndexOutOfBoundsException, dll. Pengecualian yang tidak dicentang tidak diperiksa pada waktu kompilasi melainkan diperiksa pada waktu proses.

Salah satu kasus klasik adalah ArrayIndexOutOfBoundsException yang terjadi saat Anda mencoba mengakses indeks larik yang lebih besar dari panjang larik. Berikut adalah contoh tipikal dari jenis kesalahan ini.

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (try
      (aget (int-array [1 2 3]) 5)
      (catch Exception e (println (str "caught exception: " (.toString e))))
      (finally (println "This is our final block")))
   (println "Let's move on"))
(Example)

Ketika kode di atas dijalankan, pengecualian berikut akan dimunculkan.

caught exception: java.lang.ArrayIndexOutOfBoundsException: 5
This is our final block
Let's move on

Kesalahan

Kesalahan tidak dapat dipulihkan misalnya OutOfMemoryError, VirtualMachineError, AssertionError, dll. Ini adalah kesalahan yang tidak dapat dipulihkan oleh program dan akan menyebabkan program macet. Kami sekarang membutuhkan beberapa mekanisme untuk menangkap pengecualian ini sehingga program dapat terus berjalan jika pengecualian ini ada.

Diagram berikut menunjukkan bagaimana hierarki pengecualian di Clojure diatur. Semuanya didasarkan pada hierarki yang ditentukan di Java.

Pengecualian Penangkapan

Sama seperti bahasa pemrograman lainnya, Clojure menyediakan blok 'coba-tangkap' normal untuk menangkap pengecualian saat dan kapan pengecualian itu terjadi.

Berikut ini adalah sintaks umum dari blok coba-tangkap.

(try
   (//Protected code)
   catch Exception e1)
(//Catch block)

Semua kode Anda yang dapat menimbulkan pengecualian ditempatkan di Protected code block.

Dalam catch block, Anda dapat menulis kode khusus untuk menangani pengecualian Anda sehingga aplikasi dapat memulihkan dari pengecualian tersebut.

Mari kita lihat contoh sebelumnya yang menghasilkan pengecualian file-tidak-ditemukan dan lihat bagaimana kita dapat menggunakan blok try catch untuk menangkap pengecualian yang dimunculkan oleh program.

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (try
      (def string1 (slurp "Example.txt"))
      (println string1)
      (catch Exception e (println (str "caught exception: " (.getMessage e))))))
(Example)

Program di atas menghasilkan keluaran sebagai berikut.

caught exception: Example.txt (No such file or directory)

Dari kode di atas, kami membungkus kode yang salah di file try block. Di blok catch, kami hanya menangkap pengecualian kami dan mengeluarkan pesan bahwa pengecualian telah terjadi. Jadi, kami sekarang memiliki cara yang berarti untuk menangkap pengecualian, yang dihasilkan oleh program.

Beberapa Catch Block

Seseorang dapat memiliki beberapa blok tangkapan untuk menangani berbagai jenis pengecualian. Untuk setiap blok catch, tergantung pada tipe exception yang muncul, Anda akan menulis kode untuk menanganinya.

Mari kita ubah kode sebelumnya untuk menyertakan dua blok tangkapan, satu yang khusus untuk file kita tidak ditemukan pengecualian dan yang lainnya untuk blok pengecualian umum.

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (try
      (def string1 (slurp "Example.txt"))
      (println string1)
      
      (catch java.io.FileNotFoundException e (println (str "caught file
         exception: " (.getMessage e))))
      
      (catch Exception e (println (str "caught exception: " (.getMessage e)))))
   (println "Let's move on"))
(Example)

Program di atas menghasilkan keluaran sebagai berikut.

caught file exception: Example.txt (No such file or directory)
Let's move on

Dari keluaran di atas, kita dapat dengan jelas melihat bahwa pengecualian kita ditangkap oleh blok penangkap 'FileNotFoundException' dan bukan yang umum.

Akhirnya Blokir

Blok terakhir mengikuti blok percobaan atau blok tangkap. Blok kode terakhir selalu dijalankan, terlepas dari terjadinya Exception.

Menggunakan last block memungkinkan Anda menjalankan pernyataan tipe pembersihan apa pun yang ingin Anda jalankan, apa pun yang terjadi dalam kode yang dilindungi. Berikut adalah sintaks untuk blok ini.

(try
   (//Protected code)
   catch Exception e1)
(//Catch block)
(finally
   //Cleanup code)

Mari ubah kode di atas dan tambahkan blok kode terakhir. Berikut adalah potongan kodenya.

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (try
      (def string1 (slurp "Example.txt"))
      (println string1)
      
      (catch java.io.FileNotFoundException e (println (str "caught file
         exception: " (.getMessage e))))
      
      (catch Exception e (println (str "caught exception: " (.getMessage e))))
      (finally (println "This is our final block")))
   (println "Let's move on"))
(Example)

Program di atas menghasilkan keluaran sebagai berikut.

caught file exception: Example.txt (No such file or directory)
This is our final block
Let's move on

Dari program di atas, Anda dapat melihat bahwa blok terakhir juga diterapkan setelah blok penangkap menangkap pengecualian yang diperlukan.

Karena Clojure memperoleh penanganan pengecualiannya dari Java, mirip dengan Java, metode berikut tersedia di Clojure untuk mengelola pengecualian.

  • public String getMessage()- Mengembalikan pesan rinci tentang pengecualian yang telah terjadi. Pesan ini diinisialisasi di konstruktor Throwable.

  • public Throwable getCause() - Mengembalikan penyebab pengecualian seperti yang diwakili oleh objek Throwable.

  • public String toString() - Mengembalikan nama kelas yang digabungkan dengan hasil getMessage ().

  • public void printStackTrace() - Mencetak hasil toString () bersama dengan pelacakan tumpukan ke System.err, aliran keluaran kesalahan.

  • public StackTraceElement [] getStackTrace()- Mengembalikan larik yang berisi setiap elemen pada jejak tumpukan. Elemen di indeks 0 mewakili bagian atas tumpukan panggilan, dan elemen terakhir dalam larik mewakili metode di bagian bawah tumpukan panggilan.

  • public Throwable fillInStackTrace() - Mengisi jejak tumpukan dari objek yang dapat dilempar ini dengan jejak tumpukan saat ini, menambahkan informasi sebelumnya di jejak tumpukan.

Berikut adalah contoh kode yang menggunakan beberapa metode yang tercantum di atas.

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (try
      (def string1 (slurp "Example.txt"))
      (println string1)
      
      (catch java.io.FileNotFoundException e (println (str "caught file
         exception: " (.toString e))))
      
      (catch Exception e (println (str "caught exception: " (.toString e))))
   (finally (println "This is our final block")))
   (println "Let's move on"))
(Example)

Program di atas menghasilkan keluaran sebagai berikut.

caught file exception: java.io.FileNotFoundException: Example.txt (No such file
or directory)
This is our final block
Let's move on