Clojure - Eşzamanlı Programlama

Clojure programlamada çoğu veri türü değişmezdir, bu nedenle eşzamanlı programlama söz konusu olduğunda, bu veri türlerini kullanan kod, kod birden çok işlemcide çalıştığında oldukça güvenlidir. Ancak çoğu zaman, verileri paylaşma zorunluluğu vardır ve birden çok işlemcide paylaşılan veriler söz konusu olduğunda, birden çok işlemciyle çalışırken bütünlük açısından verilerin durumunun korunmasını sağlamak gerekli hale gelir. Bu olarak bilinirconcurrent programming ve Clojure bu tür programlama için destek sağlar.

Yazılım işlemsel bellek sistemi (STM) dosync, ref, set, alter vb. Aracılığıyla açığa çıkarılır ve eşzamanlı ve koordineli bir şekilde iş parçacıkları arasında değişen durum paylaşımını destekler. Aracı sistemi, eşzamansız ve bağımsız bir şekilde iş parçacıkları arasında değişen durum paylaşımını destekler. Atom sistemi, eşzamanlı ve bağımsız bir şekilde iş parçacıkları arasında değişen durum paylaşımını destekler. Def, bağlama vb. Aracılığıyla açığa çıkan dinamik var sistemi, iş parçacıkları içindeki değişen durumu yalıtmayı destekler.

Diğer programlama dilleri de eşzamanlı programlama modelini takip eder.

  • Değiştirilebilen verilere doğrudan referansları vardır.

  • Paylaşılan erişim gerekiyorsa, nesne kilitlenir, değer değiştirilir ve bu değere bir sonraki erişim için işlem devam eder.

Clojure'da kilit yoktur, ancak değişmez kalıcı veri yapılarına dolaylı referanslar vardır.

Clojure'da üç tür referans vardır.

  • Vars - Değişiklikler ileti dizilerinde izole edilmiştir.

  • Refs - Değişiklikler iş parçacıkları arasında senkronize edilir ve koordine edilir.

  • Agents - İş parçacıkları arasında asenkron bağımsız değişiklikleri içerir.

Clojure'da eşzamanlı programlama ile ilgili olarak aşağıdaki işlemler mümkündür.

İşlemler

Clojure'da eşzamanlılık işlemlere dayanır. Referanslar yalnızca bir işlem içinde değiştirilebilir. İşlemlerde aşağıdaki kurallar uygulanır.

  • Tüm değişiklikler atomik ve izole edilmiştir.
  • Bir referansta yapılan her değişiklik bir işlemde gerçekleşir.
  • Hiçbir işlem, başka bir işlemin yaptığı etkiyi görmez.
  • Tüm işlemler dosync bloğunun içine yerleştirilir.

Dosync bloğunun ne yaptığını zaten gördük, tekrar bakalım.

Dosync

İfadeyi ve iç içe geçmiş çağrıları kapsayan bir işlemde ifadeyi (örtük bir do içinde) çalıştırır. Bu iş parçacığında zaten hiçbiri çalışmıyorsa bir işlem başlatır. Yakalanmayan herhangi bir istisna, işlemi iptal edecek ve dosync'den dışarı akacaktır.

Sözdizimi aşağıdadır.

Sözdizimi

(dosync expression)

Parameters - 'ifade', dosync bloğuna gelecek olan ifadeler kümesidir.

Return Value - Yok.

Bir referans değişkeninin değerini değiştirmeye çalıştığımız bir örneğe bakalım.

Misal

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (def names (ref []))
   (alter names conj "Mark"))
(Example)

Çıktı

Yukarıdaki program çalıştırıldığında aşağıdaki hatayı verir.

Caused by: java.lang.IllegalStateException: No transaction running
   at clojure.lang.LockingTransaction.getEx(LockingTransaction.java:208)
   at clojure.lang.Ref.alter(Ref.java:173)
   at clojure.core$alter.doInvoke(core.clj:1866)
   at clojure.lang.RestFn.invoke(RestFn.java:443)
   at clojure.examples.example$Example.invoke(main.clj:5)
   at clojure.examples.example$eval8.invoke(main.clj:7)
   at clojure.lang.Compiler.eval(Compiler.java:5424)
   ... 12 more

Hatadan, önce bir işlem başlatmadan bir referans türünün değerini değiştiremeyeceğinizi açıkça görebilirsiniz.

Yukarıdaki kodun çalışması için aşağıdaki programda yapıldığı gibi alter komutunu bir dosync bloğuna yerleştirmeliyiz.

Misal

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (def names (ref []))
   
   (defn change [newname]
      (dosync
         (alter names conj newname)))
   (change "John")
   (change "Mark")
   (println @names))
(Example)

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

Çıktı

[John Mark]

Başka bir dosync örneği görelim.

Misal

(ns clojure.examples.example
   (:gen-class))
(defn Example []
   (def var1 (ref 10))
   (def var2 (ref 20))
   (println @var1 @var2)
   
   (defn change-value [var1 var2 newvalue]
      (dosync
         (alter var1 - newvalue)
         (alter var2 + newvalue)))
   (change-value var1 var2 20)
   (println @var1 @var2))
(Example)

Yukarıdaki örnekte, bir dosync bloğunda değiştirilen iki değerimiz var. İşlem başarılı olursa, her iki değer de değişir, aksi takdirde işlemin tamamı başarısız olur.

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

Çıktı

10 20
-10 40