Clojure - Özel Durum İşleme

Exception handlinguygulamanın normal akışının sürdürülebilmesi için çalışma zamanı hatalarını işlemek için herhangi bir programlama dilinde gereklidir. İstisna genellikle uygulamanın normal akışını bozar, bu nedenle uygulamamızda istisna işlemeyi kullanmamız gerekir.

İstisna genel olarak aşağıdaki kategorilerde sınıflandırılır -

  • Checked Exception- RuntimeException ve Error dışında Throwable sınıfını genişleten sınıflar, denetlenen istisnalar olarak bilinir. Örneğin IOException, SQLException, vb. Kontrol edilen istisnalar derleme zamanında kontrol edilir.

Örnek.txt adlı bir dosya üzerinde işlem yapan aşağıdaki programı ele alalım. Bununla birlikte, Example.txt dosyasının olmadığı bir durum her zaman olabilir.

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

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

Example.txt dosyası yoksa, aşağıdaki istisna program tarafından üretilecektir.

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)

Yukarıdaki istisnadan, programın bir FileNotFoundException oluşturduğunu açıkça görebiliriz.

  • Unchecked Exception- RuntimeException'ı genişleten sınıflar, denetlenmemiş istisnalar olarak bilinir. Örneğin, ArithmeticException, NullPointerException, ArrayIndexOutOfBoundsException, vb. Denetlenmeyen istisnalar çalışma zamanında denetlenmek yerine derleme zamanında denetlenmez.

Klasik bir durum, dizinin uzunluğundan daha büyük bir dizinin dizinine erişmeye çalıştığınızda ortaya çıkan ArrayIndexOutOfBoundsException durumudur. Aşağıda bu tür bir hatanın tipik bir örneği verilmiştir.

(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)

Yukarıdaki kod çalıştırıldığında, aşağıdaki istisna ortaya çıkacaktır.

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

Hata

Hata düzeltilemez, örn. OutOfMemoryError, VirtualMachineError, AssertionError, vb. Bunlar, programın asla düzeltemeyeceği ve programın çökmesine neden olacağı hatalardır. Şimdi bu istisnaları yakalamak için bazı mekanizmalara ihtiyacımız var, böylece bu istisnalar mevcutsa program çalışmaya devam edebilir.

Aşağıdaki diyagram, Clojure'daki istisnalar hiyerarşisinin nasıl düzenlendiğini gösterir. Hepsi Java'da tanımlanan hiyerarşiye dayanmaktadır.

İstisnaları Yakalama

Tıpkı diğer programlama dillerinde olduğu gibi, Clojure istisnaları ortaya çıktıkça yakalamak için normal 'dene-yakala' bloğu sağlar.

Aşağıda, try-catch bloğunun genel sözdizimi verilmiştir.

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

Bir istisna yaratabilecek tüm kodunuz, Protected code block.

İçinde catch block, uygulamanın istisnadan kurtarabilmesi için istisnanızı işlemek için özel kod yazabilirsiniz.

Dosya bulunamadı istisnası oluşturan önceki örneğimize bakalım ve program tarafından oluşturulan istisnayı yakalamak için try catch bloğunu nasıl kullanabileceğimize bakalım.

(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)

Yukarıdaki program aşağıdaki çıktıyı üretir.

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

Yukarıdaki koddan, hatalı kodu, try block. Catch bloğunda, sadece bizim istisnamızı yakalıyoruz ve bir istisnanın meydana geldiğine dair bir mesaj çıkarıyoruz. Şimdi, program tarafından oluşturulan istisnayı yakalamanın anlamlı bir yoluna sahibiz.

Çoklu Yakalama Blokları

Birden çok istisna türünü işlemek için birden çok yakalama bloğu olabilir. Her catch bloğu için, ortaya çıkan istisna türüne bağlı olarak, onu uygun şekilde işlemek için kod yazarsınız.

Önceki kodumuzu, biri istisna bulunamadı dosyamıza özel, diğeri genel bir istisna bloğu için olan iki yakalama bloğunu içerecek şekilde değiştirelim.

(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)

Yukarıdaki program aşağıdaki çıktıyı üretir.

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

Yukarıdaki çıktıdan, istisnamızın genel değil 'FileNotFoundException' yakalama bloğu tarafından yakalandığını açıkça görebiliriz.

Sonunda Engelle

Nihayet bloğu, bir try bloğunu veya bir catch bloğunu takip eder. Son olarak bir kod bloğu, bir İstisnanın meydana gelmesinden bağımsız olarak her zaman yürütülür.

Nihayet bloğunun kullanılması, korumalı kodda ne olursa olsun, yürütmek istediğiniz herhangi bir temizleme türü ifadesini çalıştırmanıza olanak tanır. Bu bloğun sözdizimi aşağıdadır.

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

Yukarıdaki kodu değiştirelim ve en son kod bloğunu ekleyelim. Kod parçacığı aşağıdadır.

(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)

Yukarıdaki program aşağıdaki çıktıyı üretir.

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

Yukarıdaki programdan, son bloğun, catch bloğu gerekli istisnayı yakaladıktan sonra da uygulandığını görebilirsiniz.

Clojure, istisna işlemeyi Java'ya benzer şekilde Java'dan elde ettiğinden, istisnaları yönetmek için aşağıdaki yöntemler Clojure'da mevcuttur.

  • public String getMessage()- Oluşan istisna hakkında ayrıntılı bir mesaj verir. Bu mesaj Throwable yapıcısında başlatılır.

  • public Throwable getCause() - Throwable nesne tarafından temsil edildiği şekliyle istisnanın nedenini döndürür.

  • public String toString() - getMessage () sonucuyla birleştirilen sınıfın adını döndürür.

  • public void printStackTrace() - toString () sonucunu yığın izlemeyle birlikte System.err, hata çıktı akışı olarak yazdırır.

  • public StackTraceElement [] getStackTrace()- Yığın izlemesindeki her bir öğeyi içeren bir dizi döndürür. 0 dizinindeki öğe çağrı yığınının üstünü temsil eder ve dizideki son öğe çağrı yığınının altındaki yöntemi temsil eder.

  • public Throwable fillInStackTrace() - Bu Throwable nesnesinin yığın izlemesini, yığın izlemedeki önceki bilgilere ekleyerek geçerli yığın izlemesiyle doldurur.

Aşağıda, yukarıda listelenen yöntemlerden bazılarını kullanan örnek kod verilmiştir.

(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)

Yukarıdaki program aşağıdaki çıktıyı üretir.

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