Clojure - Hızlı Kılavuz
Clojure, yüksek seviyeli, dinamik bir fonksiyonel programlama dilidir. Clojure, LISP programlama diline dayalı olarak tasarlanmıştır ve hem Java hem de .Net çalışma zamanı ortamında çalışmasını sağlayan derleyicilere sahiptir.
Clojure hakkında konuşmadan önce, LISP programlama dili hakkında kısa bir açıklama yapalım. LISP'lerin küçük bir dil çekirdeği vardır, neredeyse hiç sözdizimi yoktur ve güçlü bir makro olanağı vardır. Bu özelliklerle, LISP'yi tam tersi yerine tasarımınıza uyacak şekilde bükebilirsiniz. LISP, 1958 yılına kadar uzanan uzun bir süredir oradaydı.
Common LISP bir ifadeyi okur, değerlendirir ve ardından sonucu yazdırır. Örneğin, 4 + 6'nın basit bir matematiksel ifadesinin değerini hesaplamak istiyorsanız, o zaman yazarsınız.
USER(1) (+ 4 6)
Clojure, bir programlama dili olarak aşağıdaki üst düzey temel hedeflere sahiptir.
Kod ifadelerini geleneksel programlama dillerinden daha küçük yapan LISP programlama diline dayanmaktadır.
İşlevsel bir programlama dilidir.
Temelde, yerinde oluşturulan nesnelerde herhangi bir değişiklik yapmamanız gerektiği kavramı olan değişmezliğe odaklanır.
Programcı için bir uygulamanın durumunu yönetebilir.
Eşzamanlılığı destekler.
Mevcut programlama dillerini kapsar. Örneğin Clojure, kodun JVM aracılığıyla çalıştırılmasının yönetimi için tüm Java ekosistemini kullanabilir.
Clojure için resmi web sitesi https://clojure.org/
Clojure ile programlama dili olarak çalışmanın çeşitli yolları vardır. Clojure programlamayla çalışmanın iki yolunu inceleyeceğiz.
Leiningen - Leiningen, Clojure projelerini oluşturmak, inşa etmek ve otomatikleştirmek için gerekli bir araçtır.
Eclipse Plugin - Eclipse IDE'de Clojure geliştirmeyi gerçekleştirmek için Eclipse için mevcut olan CounterClockwise adlı bir eklenti var.
Leiningen Kurulumu
Kuruluma devam etmeden önce aşağıdaki Sistem gereksinimlerinin karşılandığından emin olun.
sistem gereksinimleri
JDK | JDK 1.7 veya üstü |
---|---|
Hafıza | 2 GB RAM (önerilir) |
Step 1- İkili kurulumu indirin. Bağlantıya githttp://leiningen-wininstallerWindows Installer'ı edinmek için. Groovy yükleyiciyi indirmeye başlamak için seçeneğe tıklayın.
Step 2 - Yükleyiciyi başlatın ve İleri düğmesine tıklayın.
Step 3 - Kurulumun konumunu belirtin ve İleri düğmesine tıklayın.
Step 4- Kurulum, mevcut bir Java kurulumunun konumunu tespit edecektir. Devam etmek için İleri düğmesine tıklayın.
Step 5 - Kuruluma başlamak için Kur düğmesine tıklayın.
Kurulum tamamlandıktan sonra, size Clojure programlarınızı oluşturmak ve test etmek için kullanılabilecek bir ortam olan Clojure REPL'i açma seçeneği verecektir.
Eclipse Kurulumu
Kuruluma devam etmeden önce aşağıdaki Sistem gereksinimlerinin karşılandığından emin olun.
sistem gereksinimleri
JDK | JDK 1.7 veya üstü |
---|---|
Tutulma | Tutulma 4.5 (Mars) |
Step 1- Eclipse'i açın ve Menü öğesini tıklayın. Yardım → Eclipse Marketplace'i tıklayın.
Step 2- Görünen iletişim kutusuna Clojure anahtar kelimesini yazın ve 'Git' düğmesine basın. Saat yönünün tersine seçeneği görünecektir, bu eklentinin kurulumuna başlamak için Yükle düğmesine tıklayın.
Step 3 - Sonraki iletişim kutusunda, kuruluma başlamak için Onayla düğmesine tıklayın.
Step 4- Sonraki iletişim kutusunda, lisans sözleşmesini kabul etmeniz istenecektir. Kuruluma devam etmek için lisans sözleşmesini kabul edin ve Bitir düğmesine tıklayın.
Kurulum başlayacak ve tamamlandığında sizden Eclipse'i yeniden başlatmanızı isteyecektir.
Eclipse yeniden başlatıldığında, Eclipse'de yeni bir Clojure projesi oluşturma seçeneğini göreceksiniz.
Clojure'un temel sözdizimini anlamak için önce basit bir Merhaba Dünya programına bakalım.
Tam Bir Program Olarak Merhaba Dünya
Tam bir Clojure programına 'Merhaba dünya' yazın. Aşağıda bir örnek verilmiştir.
Misal
(ns clojure.examples.hello
(:gen-class))
(defn hello-world []
(println "Hello World"))
(hello-world)
Yukarıdaki programla ilgili olarak aşağıdaki noktalara dikkat edilmelidir.
Program main.clj adlı bir dosyaya yazılacaktır. 'Clj' uzantısı, bir clojure kod dosyasının uzantı adıdır. Yukarıdaki örnekte, dosyanın adı main.clj olarak adlandırılmıştır.
'Defn' anahtar sözcüğü bir işlevi tanımlamak için kullanılır. Fonksiyonları başka bir bölümde ayrıntılı olarak göreceğiz. Ancak şimdilik, ana Clojure kodumuza sahip olacak helloworld adlı bir işlev yarattığımızı bilin.
Clojure kodumuzda, konsol çıktısına “Merhaba Dünya” yazdırmak için 'println' ifadesini kullanıyoruz.
Daha sonra, sırayla 'println' ifadesini çalıştıran merhaba dünya işlevini çağırıyoruz.
Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
Hello World
Bir Beyanın Genel Şekli
Herhangi bir ifadenin genel biçimi aşağıdaki örnekte gösterildiği gibi parantez içinde değerlendirilmelidir.
(+ 1 2)
Yukarıdaki örnekte, ifadenin tamamı kaşlı ayraç içine alınmıştır. Yukarıdaki ifadenin çıktısı 3'tür. + İşleci, sayıların toplanması için kullanılan Clojure'da bir işlev gibi davranır. 1 ve 2'nin değerleri olarak bilinir parameters to the function.
Başka bir örneği ele alalım. Bu örnekte, 'str' iki dizeyi birleştirmek için kullanılan operatördür. "Merhaba" ve "Dünya" dizeleri parametre olarak kullanılır.
(str "Hello" "World")
Misal
Yukarıdaki iki ifadeyi birleştirip bir program yazarsak, aşağıdaki gibi görünecektir.
(ns clojure.examples.hello
(:gen-class))
(defn Example []
(println (str "Hello World"))
(println (+ 1 2)))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
Hello World
3
İsim alanları
Clojure'da tanımlanan modüller arasında mantıksal bir sınır tanımlamak için bir ad alanı kullanılır.
Mevcut Ad Alanı
Bu, mevcut Clojure kodunun içinde bulunduğu mevcut ad alanını tanımlar.
Sözdizimi
*ns*
Misal
REPL komut penceresinde aşağıdaki komutu çalıştırın.
*ns*
Çıktı
Yukarıdaki komutu çalıştırdığımızda, çıktı mevcut ad alanının ne olduğuna bağlı olarak ertelenecektir. Aşağıda bir çıktı örneği verilmiştir. Clojure kodunun ad alanı -
clojure.examples.hello
(ns clojure.examples.hello
(:gen-class))
(defn Example []
(println (str "Hello World"))
(println (+ 1 2)))
(Example)
Clojure'de Gereksinim İfadesi
Clojure kodu kitaplıklarda paketlenmiştir. Her Clojure kitaplığı, Java paketine benzer bir ad alanına aittir. Clojure kitaplığını 'Require' ifadesiyle yükleyebilirsiniz.
Sözdizimi
(require quoted-namespace-symbol)
Misal
Aşağıda bu ifadenin kullanımına bir örnek verilmiştir.
(ns clojure.examples.hello
(:gen-class))
(require ‘clojure.java.io’)
(defn Example []
(.exists (file "Example.txt")))
(Example)
Yukarıdaki kodda, girdi / çıktı işlevselliği için gerekli tüm işlevlere sahip clojure.java.io ad alanını içe aktarmak için 'require' anahtar sözcüğünü kullanıyoruz. Gerekli kitaplığa sahip olmadığımız için yukarıdaki kodda 'dosya' işlevini kullanabiliriz.
Yorumlar Clojure
Kodunuzu belgelemek için yorumlar kullanılır. Tek satır açıklamalar;; satırın herhangi bir yerinde. Aşağıda bir örnek verilmiştir.
Misal
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(println "Hello World"))
(Example)
Sınırlayıcılar
Clojure'da ifadeler, eğri veya köşeli parantezler kullanılarak bölünebilir veya sınırlandırılabilir.
Misal
Aşağıda iki örnek verilmiştir.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(println (+ 1 2 3)))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
6
Misal
Aşağıda başka bir örnek var.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(println [+ 1 2 3]))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
[#object[clojure.core$_PLUS_ 0x10f163b "clojure.core$_PLUS_@10f163b"] 1 2 3]
Beyaz boşluklar
Beyaz boşluklar, Clojure'da bir ifadenin farklı bileşenlerini daha iyi netlik için bölmek için kullanılabilir. Bu, virgül (,) operatörünün yardımıyla yapılabilir.
Örneğin, aşağıdaki iki ifade eşdeğerdir ve her iki ifadenin çıktısı 15 olacaktır.
(+ 1 2 3 4 5)
(+ 1, 2, 3, 4, 5)
Clojure virgülleri yok saysa da, bazen onları programcının okumasını kolaylaştırmak için kullanır.
Örneğin, aşağıdaki gibi bir hash haritanız varsa (def a-map {: a 1: b 2: c 3}) ve REPL penceresinde değerini sorarsanız, Clojure çıktıyı {: a 1, : b 2,: c 3}.
Özellikle büyük miktarda veriye bakıyorsanız, sonuçları okumak daha kolaydır.
Semboller
Clojure'da semboller, diğer programlama dillerindeki tanımlayıcılara eşdeğerdir. Ancak diğer programlama dillerinden farklı olarak, derleyici sembolleri gerçek dize değerleri olarak görür. Bir sembol bir değer olduğu için, bir sembol bir koleksiyonda saklanabilir, bir işleve argüman olarak iletilebilir, vb., Tıpkı diğer nesneler gibi.
Bir sembol yalnızca alfanümerik karakterler ve '* +! /. : - _? ' ancak rakam veya iki nokta üst üste ile başlamamalıdır.
Aşağıda geçerli sembol örnekleri verilmiştir.
tutorial-point!
TUTORIAL
+tutorial+
Clojure Proje Yapısı
Son olarak bir Clojure projesi için tipik bir proje yapısından bahsedelim. Clojure kodu Java sanal makinesinde çalıştığından, Clojure'deki proje yapısının çoğu bir java projesinde bulacağınızla benzerdir. Aşağıda, Clojure projesi için Eclipse'deki örnek bir proje yapısının anlık görüntüsü verilmiştir.
Yukarıdaki program yapısı hakkında aşağıdaki temel noktalara dikkat edilmelidir.
demo_1 - Clojure kod dosyasının yerleştirildiği paket budur.
core.clj - Bu, Clojure uygulamasının kodunu içerecek ana Clojure kod dosyasıdır.
Leiningen klasörü, Clojure tabanlı herhangi bir uygulamayı çalıştırmak için gereken clojure-1.6.0.jar gibi dosyaları içerir.
Pom.properties dosyası, groupId, artifactId ve Clojure projesinin sürümü gibi bilgileri içerir.
Project.clj dosyası, Clojure uygulamasının kendisi hakkında bilgi içerir. Aşağıda proje dosyası içeriğinin bir örneği bulunmaktadır.
(defproject demo-1 "0.1.0-SNAPSHOT"
:description "FIXME: write description"
:url "http://example.com/FIXME"
:license {
:name "Eclipse Public License"
:url "http://www.eclipse.org/legal/epl-v10.html"
}
:dependencies [[org.clojure/clojure "1.6.0"]])
REPL (oku-değerlendir-yazdır döngüsü), Clojure kodunu denemek için bir araçtır. Çalışan bir programla etkileşime girmenize ve işlerin olması gerektiği gibi gidip gitmediğini hızlıca denemenize olanak tanır. Bunu, size kodu girebileceğiniz bir istem sunarak yapar. Daha sonra girdinizi okur, değerlendirir, sonucu yazdırır ve tekrar bir komut istemi sunarak döngüler oluşturur.
Bu süreç, diğer birçok dilde mümkün olmayan hızlı bir geri bildirim döngüsü sağlar.
REPL Oturumu Başlatma
Leiningen'de komut satırına aşağıdaki komut yazılarak bir REPL oturumu başlatılabilir.
lein repl
Bu, aşağıdaki REPL penceresini başlatacaktır.
Daha sonra REPL penceresinde Clojure komutlarını gerektiği gibi değerlendirmeye başlarsınız.
Eclipse'de bir REPL oturumu başlatmak için Menü seçeneğine tıklayın, Farklı Çalıştır → Clojure Uygulaması'na gidin.
Bu, konsol çıktısıyla birlikte ayrı bir pencerede yeni bir REPL oturumu başlatacaktır.
REPL, kavramsal olarak Secure Shell'e (SSH) benzer. Uzak bir sunucuyla etkileşim kurmak için SSH'yi kullanabildiğiniz gibi, Clojure REPL, çalışan bir Clojure işlemiyle etkileşim kurmanıza olanak tanır. Bu özellik çok güçlü olabilir çünkü canlı bir prodüksiyon uygulamasına bir REPL bile ekleyebilir ve programınızı çalışırken değiştirebilirsiniz.
REPL'de Özel Değişkenler
REPL bazı yararlı değişkenler içerir, yaygın olarak kullanılan özel değişken * 1, * 2 ve * 3'tür. Bunlar, en son üç ifadenin sonuçlarını değerlendirmek için kullanılır.
Aşağıdaki örnek, bu değişkenlerin nasıl kullanılabileceğini göstermektedir.
user => "Hello"
Hello
user => "World"
World
user => (str *2 *1)
HelloWorld
Yukarıdaki örnekte, ilk iki dizi REPL çıktı penceresine sırasıyla "Merhaba" ve "Dünya" olarak gönderilmektedir. Daha sonra * 2 ve * 1 değişkenleri, değerlendirilen son 2 ifadeyi geri çağırmak için kullanılır.
Clojure geniş bir yelpazede built-in data types.
Yerleşik Veri Türleri
Aşağıda Clojure'da tanımlanan veri türlerinin bir listesi bulunmaktadır.
Integers - Aşağıda Clojure'da bulunan Tamsayıların temsili verilmiştir.
Decimal Integers (Short, Long and Int)- Bunlar tam sayıları temsil etmek için kullanılır. Örneğin, 1234.
Octal Numbers- Bunlar sekizlik gösterimde sayıları temsil etmek için kullanılır. Örneğin, 012.
Hexadecimal Numbers- Bunlar temsildeki sayıları temsil etmek için kullanılır. Örneğin, 0xff.
Radix Numbers- Bunlar, sayıları taban gösteriminde temsil etmek için kullanılır. Örneğin, 2r1111 burada radix 2 ile 36 arasında bir tamsayıdır.
Floating point
Varsayılan, 32 bitlik kayan nokta sayılarını temsil etmek için kullanılır. Örneğin, 12.34.
Diğer temsil, bilimsel gösterimdir. Örneğin, 1.35e-12.
char- Bu, tek bir karakter değerini tanımlar. Karakterler, boşluk simgesiyle tanımlanır. Örneğin, / e.
Boolean - Bu, doğru veya yanlış olabilen bir Boole değerini temsil eder.
String − These are text literals which are represented in the form of chain of characters. For example, “Hello World”.
Nil − This is used to represent a NULL value in Clojure.
Atom − Atoms provide a way to manage shared, synchronous, independent state. They are a reference type like refs and vars.
Bound Values
Since all of the datatypes in Clojure are inherited from Java, the bounded values are the same as in Java programming language. The following table shows the maximum allowed values for the numerical and decimal literals.
literals | Ranges |
---|---|
Short | -32,768 to 32,767 |
int | -2,147,483,648 to 2,147,483,647 |
long | -9,223,372,036,854,775,808 to +9,223,372,036,854,775,807 |
float | 1.40129846432481707e-45 to 3.40282346638528860e+38 |
double | 4.94065645841246544e-324d to 1.79769313486231570e+308d |
Class Numeric Types
In addition to the primitive types, the following object types (sometimes referred to as wrapper types) are allowed.
Name |
---|
java.lang.Byte |
java.lang.Short |
java.lang.Integer |
java.lang.Long |
java.lang.Float |
java.lang.Double |
Example
The following program shows a consolidated clojure code to demonstrate the data types in Clojure.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
;; The below code declares a integer variable
(def x 1)
;; The below code declares a float variable
(def y 1.25)
;; The below code declares a string variable
(def str1 "Hello")
(println x)
(println y)
(println str1))
(Example)
Output
The above program produces the following output.
1
1.25
Hello
In Clojure, variables are defined by the ‘def’ keyword. It’s a bit different wherein the concept of variables has more to do with binding. In Clojure, a value is bound to a variable. One key thing to note in Clojure is that variables are immutable, which means that in order for the value of the variable to change, it needs to be destroyed and recreated again.
Following are the basic types of variables in Clojure.
short − This is used to represent a short number. For example, 10.
int − This is used to represent whole numbers. For example, 1234.
long − This is used to represent a long number. For example, 10000090.
float − This is used to represent 32-bit floating point numbers. For example, 12.34.
char − This defines a single character literal. For example, ‘/a’.
Boolean − This represents a Boolean value, which can either be true or false.
String − These are text literals which are represented in the form of chain of characters. For example, “Hello World”.
Variable Declarations
Following is the general syntax of defining a variable.
Syntax
(def var-name var-value)
Where ‘var-name’ is the name of the variable and ‘var-value’ is the value bound to the variable.
Example
Following is an example of variable declaration.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
;; The below code declares a integer variable
(def x 1)
;; The below code declares a float variable
(def y 1.25)
;; The below code declares a string variable
(def str1 "Hello")
;; The below code declares a boolean variable
(def status true))
(Example)
Naming Variables
The name of a variable can be composed of letters, digits, and the underscore character. It must begin with either a letter or an underscore. Upper and lowercase letters are distinct because Clojure, just like Java is a case-sensitive programming language.
Example
Following are some examples of variable naming in Clojure.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
;; The below code declares a Boolean variable with the name of status
(def status true)
;; The below code declares a Boolean variable with the name of STATUS
(def STATUS false)
;; The below code declares a variable with an underscore character.
(def _num1 2))
(Example)
Note − In the above statements, because of the case sensitivity, status and STATUS are two different variable defines in Clojure.
The above example shows how to define a variable with an underscore character.
Printing variables
Since Clojure uses the JVM environment, you can also use the ‘println’ function. The following example shows how this can be achieved.
Example
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
;; The below code declares a integer variable
(def x 1)
;; The below code declares a float variable
(def y 1.25)
;; The below code declares a string variable
(def str1 "Hello")
(println x)
(println y)
(println str1))
(Example)
Output
The above program produces the following output.
1
1.25
Hello
An operator is a symbol that tells the compiler to perform specific mathematical or logical manipulations.
Clojure has the following types of operators −
- Arithmetic operators
- Relational operators
- Logical operators
- Bitwise operators
Note − In Clojure, operators and operands work in the following syntax manner.
Syntax
(operator operand1 operand2 operandn)
For example,
Example
(+ 1 2)
The above example does an arithmetic operation on the numbers 1 and 2.
Arithmetic Operators
Clojure language supports the normal Arithmetic operators as any language. Following are the Arithmetic operators available in Clojure.
Show Examples
Operator | Description | Example |
---|---|---|
+ | Addition of two operands | (+ 1 2) will give 3 |
− | Subtracts second operand from the first | (- 2 1) will give 1 |
* | Multiplication of both operands | (* 2 2) will give 4 |
/ | Division of numerator by denominator | (float (/ 3 2)) will give 1.5 |
inc | Incremental operators used to increment the value of an operand by 1 | inc 5 will give 6 |
dec | Incremental operators used to decrement the value of an operand by 1 | dec 5 will give 4 |
max | Returns the largest of its arguments | max 1 2 3 will return 3 |
min | Returns the smallest of its arguments | min 1 2 3 will return 1 |
rem | Remainder of dividing the first number by the second | rem 3 2 will give 1 |
Relational Operators
Relational operators allow comparison of objects. Following are the relational operators available in Clojure.
Show Examples
Operator | Description | Example |
---|---|---|
= | Tests the equality between two objects | (= 2 2) will give true |
not= | Tests the difference between two objects | (not = 3 2) will give true |
< | Checks to see if the left object is less than the right operand | (< 2 3) will give true |
<= | Checks to see if the left object is less than or equal to the right operand | (<= 2 3) will give true |
> | Checks to see if the left object is greater than the right operand | (> 3 2) will give true |
>= | Checks to see if the left object is greater than or equal to the right operand | (>= 3 2) will give true |
Logical Operators
Logical operators are used to evaluate Boolean expressions. Following are the logical operators available in Groovy.
Show Examples
Operator | Description | Example |
---|---|---|
and | This is the logical “and” operator | (or true true) will give true |
or | This is the logical “or” operator | (and true false) will give false |
not | This is the logical “not” operator | (not false) will give true |
The following code snippet shows how the various operators can be used.
Bitwise Operators
Clojure provides four bitwise operators. Following are the bitwise operators available in Clojure.
Show Examples
Sr.No. | Operator & Description |
---|---|
1 | bit-and This is the bitwise “and” operator |
2 | bit-or This is the bitwise “or” operator |
3 | bit-xor This is the bitwise “xor” or Exclusive ‘or’ operator |
4 | bit-not This is the bitwise negation operator |
Following is the truth table showcasing these operators.
p | q | p&q | p | q | p ^ q |
---|---|---|---|---|
0 | 0 | 0 | 0 | 0 |
0 | 1 | 0 | 1 | 1 |
1 | 1 | 1 | 1 | 0 |
1 | 0 | 0 | 1 | 1 |
Operator Precedence
As is the case with LISPs in general, there is no need to worry about operator precedence. This is one of the benefits of S-Expressions and prefix notation. All functions evaluate left to right and inside out. The operators in Clojure are just functions, and everything is fully parenthesized.
So far we have seen statements which are executed one after the other in a sequential manner. Additionally, statements are provided in Clojure to alter the flow of control in a program’s logic. They are then classified into flow of control statements which we will see in detail.
Sr.No. | Loops & Description |
---|---|
1 | While Statement The 'while' statement is executed by first evaluating the condition expression (a Boolean value), and if the result is true, then the statements in the while loop are executed. |
2 | Doseq Statement The ‘doseq’ statement is similar to the ‘for each’ statement which is found in many other programming languages. The doseq statement is basically used to iterate over a sequence. |
3 | Dotimes Statement The ‘dotimes’ statement is used to execute a statement ‘x’ number of times. |
4 | Loop Statement The loop special form is not like a ‘for’ loop. The usage of loop is the same as the let binding. However, loop sets a recursion point |
Decision-making structures require that the programmer specify one or more conditions to be evaluated or tested by the program, along with a statement or statements to be executed if the condition is determined to be true, and optionally, other statements to be executed if the condition is determined to be false.
Sr.No. | Methods & Description |
---|---|
1 | If Statement In Clojure, the condition is an expression which evaluates it to be either true or false. 'If' the condition is true, then statement#1 will be executed, else statement#2 will be executed. |
2 | If/do Expression The ‘if-do’ expression in Clojure is used to allow multiple expressions to be executed for each branch of the ‘if’ statement. |
3 | Nested If Statement Multiple 'if' statements embedded inside each other. |
4 | Case Statement Clojure offers the ‘case’ statement which is similar to the ‘switch’ statement available in the Java programming language. |
5 | Cond Statement Clojure offers another evaluation statement called the ‘cond’ statement. This statement takes a set of test/expression pairs. |
Clojure is known as a functional programming language, hence you would expect to see a lot of emphasis on how functions work in Clojure. This chapter covers what all can be done with functions in Clojure.
Sr.No. | Functions & Description |
---|---|
1 | Defining a Function A function is defined by using the ‘defn’ macro. |
2 | Anonymous Functions An anonymous function is a function which has no name associated with it. |
3 | Functions with Multiple Arguments Clojure functions can be defined with zero or more parameters. The values you pass to functions are called arguments, and the arguments can be of any type. |
4 | Variadic Functions Clojure offers the ‘case’ statement which is similar to the ‘switch’ statement available in the Java programming language. |
5 | Higher Order Functions Higher-order functions (HOFs) are functions that take other functions as arguments. HOFs are an important functional programming technique and are quite commonly used in Clojure. |
Numbers datatype in Clojure is derived from Java classes.
Clojure supports integer and floating point numbers.
An integer is a value that does not include a fraction.
A floating-point number is a decimal value that includes a decimal fraction.
Following is an example of numbers in Clojure.
(def x 5)
(def y 5.25)
Where ‘x’ is of the type Integer and ‘y’ is the float.
In Java, the following classes are attached to the numbers defined in Clojure.
To actually see that the numbers in Clojure are derived from Java classes, use the following program to see the type of numbers assigned when using the ‘def’ command.
Example
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(def x 5)
(def y 5.25)
(println (type x))
(println (type y)))
(Example)
The ‘type’ command is used to output the class associated with the value assigned to a variable.
Output
The above code will produce the following output.
Java.lang.long
Java.lang.double
Number Tests
The following test functions are available for numbers.
Sr.No. | Numbers & Description |
---|---|
1 | zero? Returns true if the number is zero, else false. |
2 | pos? Returns true if number is greater than zero, else false. |
3 | neg? Returns true if number is less than zero, else false. |
4 | even? Returns true if the number is even, and throws an exception if the number is not an integer. |
5 | odd? Returns true if the number is odd, and throws an exception if the number is not an integer. |
6 | number? Returns true if the number is really a Number. |
7 | integer? Returns true if the number is an integer. |
8 | float? Returns true if the number is a float. |
We have seen the recur statement in an earlier topic and whereas the ‘for’ loop is somewhat like a loop, recur is a real loop in Clojure.
If you have a programming background, you may have heard of tail recursion, which is a major feature of functional languages. This recur special form is the one that implements tail recursion. As the word “tail recursion” indicates, recur must be called in the tail position. In other words, recur must be the last thing to be evaluated.
The simplest example of the recur statement is used within the ‘for’ loop. In the following example, the recur statement is used to change the value of the variable ‘i’ and feed the value of the variable back to the loop expression.
Example
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(loop [i 0]
(when (< i 5)
(println i)
(recur (inc i)))))
(Example)
Output
The above program produces the following output.
0
1
2
3
4
Clojure provides a number of helper methods when working with I/O. It offers easier classes to provide the following functionalities for files.
- Reading files
- Writing to files
- Seeing whether a file is a file or directory
Let’s explore some of the file operations Clojure has to offer.
Reading the Contents of a File as an Entire String
If you want to get the entire contents of the file as a string, you can use the clojure.core.slurp method. The slurp command opens a reader on a file and reads all its contents, returning a string.
Following is an example of how this can be done.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(def string1 (slurp "Example.txt"))
(println string1))
(Example)
If the file contains the following lines, they will be printed as −
line : Example1
line : Example2
Reading the Contents of a File One Line at a Time
If you want to get the entire contents of the file as a string one line at a time, you can use the clojure.java.io/reader method. The clojure.java.io/reader class creates a reader buffer, which is used to read each line of the file.
Following is an example that shows how this can be done.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(with-open [rdr (clojure.java.io/reader "Example.txt")]
(reduce conj [] (line-seq rdr))))
(Example)
If the file contains the following lines, they will be printed as −
line : Example1
line : Example2
The output will be shown as −
["line : Example1" "line : Example2"]
Writing ‘to’ Files
If you want to write ‘to’ files, you can use the clojure.core.spit command to spew entire strings into files. The spit command is the opposite of the slurp method. This method opens a file as a writer, writes content, then closes file.
Following is an example.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(spit "Example.txt"
"This is a string"))
In the above example, if you see the contents of the Example.txt file , you will see the contents of “This is a string”.
Writing ‘to’ Files One Line at a Time
If you want to write ‘to’ files one line at a time, you can use the clojure.java.io.writer class. The clojure.java.io.writer class is used to create a writer stream wherein bytes of data are fed into the stream and subsequently into the file.
Following is an example that shows how the spit command can be used.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(with-open [w (clojure.java.io/writer "Example.txt" :append true)]
(.write w (str "hello" "world"))))
(Example)
When the above code is executed, the line “hello world” will be present in the Example.txt file. The append:true option is to append data to the file. If this option is not specified, then the file will be overwritten whenever data is written to the file.
Checking to See If a File Exists
To check if a file exists, the clojure.java.io.file class can be used to check for the existence of a file. Following is an example that shows how this can be accomplished.
(ns clojure.examples.hello
(:gen-class))
;; This program displays Hello World
(defn Example []
(println (.exists (clojure.java.io/file "Example.txt"))))
(Example)
If the file Example.txt exists, the output will be true.
Reading from the Console
To read data from the console, the read-line statement can be used. Following is an example that shows how this can be used.
If you enter the (read-line) command in the REPL window, you will have the chance to enter some input in the console window.
user->(read-line)
Hello World
The above code will produce the following output.
“Hello World”
A String literal is constructed in Clojure by enclosing the string text in quotations. Strings in Clojure need to be constructed using the double quotation marks such as “Hello World”.
Example
Following is an example of the usage of strings in Clojure.
(ns clojure.examples.hello
(:gen-class))
(defn hello-world []
(println "Hello World")
(println "This is a demo application"))
(hello-world)
Output
The above program produces the following output.
Hello World
This is a demo application
Basic String Operations
Clojure has a number of operations that can be performed on strings. Following are the operations.
Sr.No. | String Operations & Description |
---|---|
1 | str The concatenation of strings can be done by the simple str function. |
2 | format The formatting of strings can be done by the simple format function. The format function formats a string using java.lang.String.format. |
3 | count Returns the number of characters in the string. |
4 | subs Returns the substring of ‘s’ beginning at start inclusive, and ending at end (defaults to length of string), exclusive. |
5 | compare Returns a negative number, zero, or a positive number when ‘x’ is logically 'less than', 'equal to', or 'greater than' ‘y’. |
6 | lower-case Converts string to all lower-case. |
7 | upper-case Converts string to all upper-case. |
8 | join Returns a string of all elements in collection, as returned by (seq collection), separated by an optional separator. |
9 | split Splits string on a regular expression. |
10 | split-lines Split strings is based on the escape characters \n or \r\n. |
11 | reverse Reverses the characters in a string. |
12 | replace Replaces all instance of a match in a string with the replacement string. |
13 | trim Removes whitespace from both ends of the string. |
14 | triml Removes whitespace from the left hand side of the string. |
15 | trimr Removes whitespace from the right hand side of the string. |
List is a structure used to store a collection of data items. In Clojure, the List implements the ISeq interface. Lists are created in Clojure by using the list function.
Example
Following is an example of creating a list of numbers in Clojure.
(ns clojure.examples.example
(:gen-class))
(defn example []
(println (list 1 2 3 4)))
(example)
Output
The above code produces the following output.
(1 2 3 4)
Following is an example of creating a list of characters in Clojure.
(ns clojure.examples.example
(:gen-class))
(defn example []
(println (list 'a 'b 'c 'd)))
(example)
The above code produces the following output.
(a b c d)
Following are the list methods available in Clojure.
Sr.No. | Lists & Description |
---|---|
1 | list* Creates a new list containing the items prepended to the rest, the last of which will be treated as a sequence. |
2 | first This function returns the first item in the list. |
3 | nth This function returns the item in the ‘nth’ position in the list. |
4 | cons Returns a new list wherein an element is added to the beginning of the list. |
5 | conj Returns a new list wherein the list is at the beginning and the elements to be appended are placed at the end. |
6 | rest Returns the remaining items in the list after the first item. |
Sets in Clojure are a set of unique values. Sets are created in Clojure with the help of the set command.
Example
Following is an example of the creation of sets in Clojure.
(ns clojure.examples.example
(:gen-class))
(defn example []
(println (set '(1 1 2 2))))
(example)
Output
The above code produces the following output.
#{1,2}
Following are the methods available in Clojure for sets.
Sr.No. | Sets & Description |
---|---|
1 | sorted-set Returns a sorted set of elements. |
2 | get Returns the element at the index position. |
3 | contains? Finds out whether the set contains a certain element or not. |
4 | conj Appends an element to the set and returns the new set of elements. |
5 | disj Disjoins an element from the set. |
6 | union Return a set that is the union of the input sets |
7 | difference Return a set that is the first set without elements of the remaining sets. |
8 | intersection Return a set that is the intersection of the input sets. |
9 | subset? Is set1 a subset of set2? |
10 | superset? Is set1 a superset of set2? |
A Vector is a collection of values indexed by contiguous integers. A vector is created by using the vector method in Clojure.
Example
Following is an example of creating a vector in Clojure.
(ns clojure.examples.example
(:require [clojure.set :as set])
(:gen-class))
(defn example []
(println (vector 1 2 3)))
(example)
Output
The above code produces the following output.
[1 2 3]
Following are the methods available in Clojure.
Sr.No. | Vectors & Description |
---|---|
1 | vector-of Creates a new vector of a single primitive type ‘t’, where ‘t’ is one of :int :long :float :double :byte :short :char or :boolean. |
2 | nth This function returns the item in the nth position in the vector. |
3 | get Returns the element at the index position in the vector. |
4 | conj Appends an element to the vector and returns the new set of vector elements. |
5 | pop For a list or queue, returns a new list/queue without the first item, for a vector, returns a new vector without the last item. |
6 | subvec Returns a sub vector from a starting and ending index. |
A Map is a collection that maps keys to values. Two different map types are provided - hashed and sorted. HashMaps require keys that correctly support hashCode and equals. SortedMaps require keys that implement Comparable, or an instance of Comparator.
A map can be created in two ways, the first is via the hash-map method.
Creation - HashMaps
HashMaps have a typical key value relationship and is created by using hash-map function.
(ns clojure.examples.example
(:gen-class))
(defn example []
(def demokeys (hash-map "z" "1" "b" "2" "a" "3"))
(println demokeys))
(example)
Output
The above code produces the following output.
{z 1, b 2, a 3}
Creation - SortedMaps
SortedMaps have the unique characteristic of sorting their elements based on the key element. Following is an example that shows how the sorted map can be created using the sorted-map function.
(ns clojure.examples.example
(:gen-class))
(defn example []
(def demokeys (sorted-map "z" "1" "b" "2" "a" "3"))
(println demokeys))
(example)
The above code produces the following output.
{a 3, b 2, z 1}
From the above program you can clearly see that elements in the maps are sorted as per the key value. Following are the methods available for maps.
Sr.No. | Maps & Description |
---|---|
1 | get Returns the value mapped to key, not-found or nil if key is not present. |
2 | contains? See whether the map contains a required key. |
3 | find Returns the map entry for the key. |
4 | keys Returns the list of keys in the map. |
5 | vals Returns the list of values in the map. |
6 | dissoc Dissociates a key value entry from the map. |
7 | merge Merges two maps entries into one single map entry. |
8 | merge-with Returns a map that consists of the rest of the maps conj-ed onto the first. |
9 | select-keys Returns a map containing only those entries in map whose key is in keys. |
10 | rename-keys Renames keys in the current HashMap to the newly defined ones. |
11 | map-invert Inverts the maps so that the values become the keys and vice versa. |
Namespaces in Clojure are used to differentiate classes into separate logical spaces just like in Java. Consider the following statement.
(:require [clojure.set :as set])
In the above statement, ‘clojure.set’ is a namespace which contains various classes and methods to be used in the program. For example, the above namespace contains the function called map-invert, which is used to invert a map of key-values. We cannot use this function unless we explicitly tell our program to include this namespace.
Let’s look at the different methods available for namespaces.
Sr.No. | Methods & Description |
---|---|
1 | *ns* This is used to look at your current namespace. |
2 | ns This is used to create a new namespace and associate it with the running program. |
3 | alias Add an alias in the current namespace to another namespace. Arguments are two symbols: the alias to be used and the symbolic name of the target namespace. |
4 | all-ns Returns a list of all namespaces. |
5 | find-ns Finds and returns a particular namespace. |
6 | ns-name Returns the name of a particular namespace. |
7 | ns-aliases Returns the aliases, which are associated with any namespaces. |
8 | ns-map Returns a map of all the mappings for the namespace. |
9 | un-alias Returns a map containing only those entries in map whose key is in keys. |
Exception handling is required in any programming language to handle the runtime errors so that the normal flow of the application can be maintained. Exception usually disrupts the normal flow of the application, which is the reason why we need to use exception handling in our application.
Exception is broadly classified into the following categories −
Checked Exception − The classes that extend Throwable class except RuntimeException and Error are known as checked exceptions. E.g. IOException, SQLException, etc. Checked exceptions are checked at compile-time.
Let’s consider the following program which does an operation on a file called Example.txt. However, there can be always a case wherein the file Example.txt does not exist.
(ns clojure.examples.example
(:gen-class))
;; This program displays Hello World
(defn Example []
(def string1 (slurp "Example.txt"))
(println string1))
(Example)
If the file Example.txt does not exist, then the following exception will be generated by the 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)
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 olan bir dizinin dizinine erişmeye çalıştığınızda gerçekleşen 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, örneğin 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 ve ortaya çıktıklarında 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 oluşturabilecek tüm kodunuz, Protected code block.
İçinde catch block, uygulamanın istisnadan kurtarabilmesi için istisnanızı işlemek için özel kod yazabilirsiniz.
Bir 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ğimizi görelim.
(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 fazla 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 iki catch bloğu içerecek şekilde değiştirelim; biri istisna bulunamadı dosyamıza özel, diğeri ise genel bir istisna bloğu için.
(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. Nihayetinde bir kod bloğu, bir İstisnanın ortaya çıkmasına bakılmaksızın 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 Clojure'da aşağıdaki yöntemler 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
Sequences yardımıyla oluşturulur ‘seq’komut. Aşağıda, bir dizi oluşturmanın basit bir örneği verilmiştir.
(ns clojure.examples.example
(:gen-class))
;; This program displays Hello World
(defn Example []
(println (seq [1 2 3])))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir.
(1 2 3)
Aşağıda, diziler için kullanılabilen çeşitli yöntemler yer almaktadır.
Sr.No. | Yöntemler ve Açıklama |
---|---|
1 | Eksileri "X" in ilk öğe ve "seq" geri kalanı olduğu yeni bir dizi döndürür. |
2 | birleşik Dizinin sonuna eklenen "x" öğesinin olduğu yeni bir dizi döndürür. |
3 | concat Bu, iki diziyi bir araya getirmek için kullanılır. |
4 | farklı Yalnızca sıraya farklı öğelerin eklendiğinden emin olmak için kullanılır. |
5 | tersine çevirmek Sıradaki öğeleri ters çevirir. |
6 | ilk Dizinin ilk öğesini döndürür. |
7 | son Dizinin son öğesini verir. |
8 | dinlenme İlk öğe dışındaki dizinin tamamını döndürür. |
9 | çeşit Sıralanmış bir öğe dizisi döndürür. |
10 | düşürmek Bir dizideki öğeleri, kaldırılması gereken öğelerin sayısına göre düşürür. |
11 | son alma Dizideki son öğe listesini alır. |
12 | almak Dizideki ilk öğe listesini alır. |
13 | bölme Öğe sırasını iki bölüme ayırır. Bölünmenin gerçekleşmesi gereken bir konum belirtilir. |
Bir regular expressionmetinde alt dizeleri bulmak için kullanılan bir kalıptır. Düzenli ifadeler çeşitli programlama dillerinde kullanılır ve LISP tipi programlama dillerinde çokça kullanılır.
Aşağıda bir normal ifade örneği verilmiştir.
//d+
Yukarıdaki normal ifade, bir dizedeki bir basamağın bir kez daha tekrarını bulmak için kullanılır. // karakterleri, 'd' ve '+' karakterlerinin bir normal ifadeyi temsil etmek için kullanıldığından emin olmak için kullanılır.
Genel olarak, normal ifadeler aşağıdaki kural kümesiyle çalışır.
Bir satırın başlangıcını ve sonunu belirtmek için kullanılan iki özel konumsal karakter vardır: düzeltme işareti (∧) ve dolar işareti ($):
Normal ifadeler ayrıca nicelik belirteçleri içerebilir. Artı işareti (+), ifadenin önceki öğesine uygulanan bir veya daha fazla zamanı temsil eder. Yıldız işareti (*), sıfır veya daha fazla oluşumu temsil etmek için kullanılır. Soru işareti (?) Sıfırı veya bir defa gösterir.
Meta karakter {ve}, önceki karakterin belirli sayıda örneğini eşleştirmek için kullanılır.
Normal ifadede nokta sembolü (.) Herhangi bir karakteri temsil edebilir. Bu, joker karakter olarak tanımlanır.
Normal bir ifade, karakter sınıflarını içerebilir. Bir dizi karakter, [aeiou] 'da olduğu gibi meta karakterler [ve] içine alınmış basit bir karakter dizisi olarak verilebilir. Harf veya sayı aralıkları için, [a – z] veya [a – mA – M] 'de olduğu gibi bir tire ayırıcısı kullanabilirsiniz. Bir karakter sınıfının tamamlayıcısı, [∧a – z] 'de olduğu gibi köşeli parantezler içinde baştaki bir düzeltme işareti ile gösterilir ve belirtilenler dışındaki tüm karakterleri temsil eder.
Normal ifadeler için aşağıdaki yöntemler mevcuttur.
Sr.No. | Yöntemler ve Açıklama |
---|---|
1 | yeniden desen Java.util.regex.Pattern örneğini döndürür. Bu daha sonra desen eşleştirme için diğer yöntemlerde kullanılır. |
2 | gözden geçirmek Java.util.regex.Matcher.find () kullanarak dizeye dizeden sonraki düzenli ifade eşleşmesini (varsa) döndürür |
3 | yerine koymak Replace işlevi, bir dizedeki bir alt dizeyi yeni bir dize değeriyle değiştirmek için kullanılır. Alt dize için arama, bir desen kullanılarak yapılır. |
4 | önce değiştir Replace işlevi, bir dizedeki bir alt dizeyi yeni bir dize değeriyle değiştirmek için kullanılır, ancak yalnızca alt dizenin ilk oluşumu için kullanılır. Alt dize için arama, bir desen kullanılarak yapılır. |
PredicatesBir koşulu değerlendiren ve doğru veya yanlış değeri sağlayan işlevlerdir. Sayılarla ilgili bölümdeki örneklerde yüklem fonksiyonlarını gördük. "Çift" gibi işlevler gördük. bir sayının çift olup olmadığını veya 'neg?' olup olmadığını test etmek için hangisi kullanılır? Bu, bir sayının sıfırdan büyük olup olmadığını test etmek için kullanılır. Bu işlevlerin tümü, doğru veya yanlış bir değer döndürür.
Aşağıda Clojure'deki yüklemlere bir örnek verilmiştir.
(ns clojure.examples.example
(:gen-class))
;; This program displays Hello World
(defn Example []
(def x (even? 0))
(println x)
(def x (neg? 2))
(println x)
(def x (odd? 3))
(println x)
(def x (pos? 3))
(println x))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir.
true
false
true
true
Normal yüklem işlevlerine ek olarak, Clojure yüklemler için daha fazla işlev sağlar. Tahminler için aşağıdaki yöntemler mevcuttur.
Sr.No. | Yöntemler ve Açıklama |
---|---|
1 | her zaman Bir dizi yüklemi alır ve oluşturan tüm yüklemleri tüm bağımsız değişkenlerine karşı mantıksal bir gerçek değer döndürürse doğru döndüren bir 'f' işlevi döndürür, aksi takdirde yanlış döndürür. |
2 | her? Yüklem her değer için doğruysa doğru, aksi takdirde yanlış döndürür. |
3 | biraz Değerler koleksiyonundaki x'in herhangi bir yüklem değeri için ilk mantıksal doğru değeri döndürür. |
4 | Hiç? Bir koleksiyondaki değerlerin herhangi bir koşulu mantıksal olarak doğruysa yanlış, aksi takdirde doğru döndürür. |
Destructuring Clojure içindeki bir işlevselliktir, bir vektör gibi bir veri yapısından değerlerin çıkarılmasına ve veri yapısını açık bir şekilde geçmek zorunda kalmadan bunları sembollere bağlamasına izin verir.
Yıkımın tam olarak ne anlama geldiğine ve nasıl gerçekleştiğine dair bir örneğe bakalım.
Misal
(ns clojure.examples.example
(:gen-class))
(defn Example []
(def my-vector [1 2 3 4])
(let [[a b c d] my-vector]
(println a b c d)))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
1 2 3 4
Yukarıdaki örnekte, aşağıdaki noktalara dikkat edilmelidir -
Tam sayılardan oluşan bir vektörü 1, 2, 3 ve 4 olarak tanımlıyoruz.
Daha sonra kullanıyoruz ‘let’ 4 değişkeni (a, b, c ve d) my-vector değişkenine doğrudan atamak için ifade.
Eğer koşarsak ‘println’ dört değişkenle ilgili ifadede, bunların sırasıyla vektördeki değerlere atanmış olduklarını görebiliriz.
Bu yüzden clojure, 'let' ifadesi kullanılarak atandığında dört değeri olan my-vector değişkenini yok etti. Yapısı bozulmuş dört değer daha sonra buna göre dört parametreye atandı.
Atanabilecekleri karşılık gelen bir değere sahip olmayan fazla değişkenler varsa, bunlara sıfır değeri atanacaktır. Aşağıdaki örnek bu noktayı netleştirir.
Misal
(ns clojure.examples.hello
(:gen-class))
(defn Example []
(def my-vector [1 2 3 4])
(let [[a b c d e] my-vector]
(println a b c d e)))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir. Son değişken 'e' vektörde karşılık gelen bir değere sahip olmadığından, bunun sıfır olduğunu çıktıdan görebilirsiniz.
Çıktı
1 2 3 4 nil
geri kalan
'The-rest' değişkeni, herhangi bir değişkene atanamayan kalan değerleri saklamak için kullanılır.
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
Misal
(ns clojure.examples.example
(:gen-class))
(defn Example []
(def my-vector [1 2 3 4])
(let [[a b & the-rest] my-vector]
(println a b the-rest)))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir. Çıktıdan, 3 ve 4 değerlerinin herhangi bir değişkene atanamayacağını, dolayısıyla 'geri kalan' değişkenine atandıklarını açıkça görebilirsiniz.
Çıktı
1 2 (3 4)
Haritaları Yıkmak
Vektörler gibi haritalar da imha edilebilir. Aşağıda bunun nasıl başarılabileceğine dair bir örnek verilmiştir.
Misal
(ns clojure.examples.example
(:gen-class))
(defn Example []
(def my-map {"a" 1 "b" 2})
(let [{a "a" b "b"} my-map]
(println a b)))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir. Programdan, “a” ve “b” nin harita değerlerinin a ve b değişkenlerine atandığını açıkça görebilirsiniz.
Çıktı
1 2
Benzer şekilde, vektörler durumunda, yok etme meydana geldiğinde haritada karşılık gelen bir değer yoksa, değişkene bir sıfır değeri atanacaktır.
Aşağıda bir örnek verilmiştir.
Misal
(ns clojure.examples.example
(:gen-class))
(defn Example []
(def my-map {"a" 1 "b" 2})
(let [{a "a" b "b" c "c"} my-map]
(println a b c)))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
1 2 nil
Clojure çerçevesi Java sınıflarından türetildiği için, Clojure'da Java'da bulunan tarih-saat sınıfları kullanılabilir. class date milisaniye hassasiyetiyle belirli bir anı temsil eder.
Tarih-saat sınıfı için kullanılabilen yöntemler aşağıdadır.
java.util.Date
Bu, Clojure'da tarih nesnesini oluşturmak için kullanılır.
Sözdizimi
Sözdizimi aşağıdadır.
java.util.Date.
Parameters - Yok.
Return Value - Bir Date nesnesini ayırır ve onu en yakın milisaniye ile ölçülen, tahsis edildiği zamanı temsil edecek şekilde başlatır.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns example)
(defn Example []
(def date (.toString (java.util.Date.)))
(println date))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir. Bu, programın çalıştırıldığı sistemdeki geçerli tarih ve saate bağlı olacaktır.
Tue Mar 01 06:11:17 UTC 2016
java.text.SimpleDateFormat
Bu, tarih çıktısını biçimlendirmek için kullanılır.
Sözdizimi
Sözdizimi aşağıdadır.
(java.text.SimpleDateFormat. format dt)
Parameters- 'biçim', tarihi biçimlendirirken kullanılacak biçimdir. 'dt' biçimlendirilmesi gereken tarihtir.
Return Value - Biçimlendirilmiş bir tarih çıktısı.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns example)
(defn Example []
(def date (.format (java.text.SimpleDateFormat. "MM/dd/yyyy") (new java.util.Date)))
(println date))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir. Bu, programın çalıştırıldığı sistemdeki geçerli tarih ve saate bağlı olacaktır.
03/01/2016
getTime
Bu Date nesnesi tarafından temsil edilen 1 Ocak 1970 00:00:00 GMT'den bu yana geçen milisaniye sayısını döndürür.
Sözdizimi
Sözdizimi aşağıdadır.
(.getTime)
Parameters - Yok.
Return Value - 1 Ocak 1970 00:00:00 GMT'den bu yana bu tarihle gösterilen milisaniye sayısı.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns example)
(import java.util.Date)
(defn Example []
(def date (.getTime (java.util.Date.)))
(println date))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir. Bu, programın çalıştırıldığı sistemdeki geçerli tarih ve saate bağlı olacaktır.
1456812778160
AtomsClojure'da paylaşılan, eşzamanlı, bağımsız durumu yönetmenin bir yolunu sağlayan bir veri türüdür. Bir atom, diğer herhangi bir programlama dilindeki herhangi bir referans türü gibidir. Bir atomun birincil kullanımı, Clojure'un değişmez veri yapılarını tutmaktır. Bir atomun tuttuğu değer,swap! method.
Dahili olarak, takas! geçerli değeri okur, işlevi ona uygular ve onu karşılaştırmaya ve ayarlamaya çalışır. Aradaki zamanda başka bir iş parçacığı değeri değiştirmiş olabileceğinden, yeniden denemek zorunda kalabilir ve bunu bir döndürme döngüsünde yapar. Net etki, değerin her zaman sağlanan fonksiyonun atomik olarak bir akım değerine uygulanmasının sonucu olacaktır.
Misal
Atom yöntemi yardımıyla atomlar oluşturulur. Aşağıdaki programda bununla ilgili bir örnek gösterilmektedir.
(ns clojure.examples.example
(:gen-class))
(defn example []
(def myatom (atom 1))
(println @myatom))
(example)
Çıktı
Yukarıdaki program aşağıdaki sonucu verir.
1
@ Sembolü kullanılarak atomun değerine erişilir. Clojure'un atomlar üzerinde yapılabilen birkaç işlemi vardır. İşlemler aşağıdadır.
Sr.No. | İşlemler ve Açıklama |
---|---|
1 | Sıfırla! Mevcut değere bakılmaksızın atom değerini yeni bir değere ayarlar. |
2 | karşılaştır ve ayarla! Atomun değerini yeni değere atomik olarak ayarlar, ancak ve ancak atomun mevcut değeri, atomun tuttuğu eski değerle aynıysa. Set olursa doğru, aksi takdirde yanlış döndürür. |
3 | takas! Atomun değerini, belirli bir işleve dayalı olarak yenisiyle atomik olarak değiştirir. |
Clojure'da, metadatabir koleksiyondaki verilere veya bir sembolde depolanan verilere açıklama eklemek için kullanılır. Bu normalde temeldeki derleyiciye türler hakkındaki verileri açıklamak için kullanılır, ancak geliştiriciler için de kullanılabilir. Meta veriler, nesnenin değerinin bir parçası olarak kabul edilmez. Aynı zamanda, meta veriler değişmezdir.
Clojure'da meta verilerle ilgili olarak aşağıdaki işlemler mümkündür.
Sr.No. | İşlemler ve Açıklama |
---|---|
1 | meta-with Bu işlev, herhangi bir nesne için bir meta veri eşlemesi tanımlamak için kullanılır. |
2 | meta Bu işlev, herhangi bir meta verinin bir nesneyle ilişkilendirilip ilişkilendirilmediğini görmek için kullanılır. |
3 | değişken meta Orijinal nesneyle aynı tür ve değere sahip, ancak birleşik meta veriler içeren bir nesne döndürür. |
StructMapsClojure'da yapılar oluşturmak için kullanılır. Örneğin, bir Employee Name ve Employeeid'den oluşan bir yapı oluşturmak istiyorsanız, bunu StructMaps ile yapabilirsiniz.
Clojure'da StructMaps ile ilgili olarak aşağıdaki işlemler mümkündür.
Sr.No. | İşlemler ve Açıklama |
---|---|
1 | kusurlu Bu fonksiyon, gerekli yapıyı tanımlamak için kullanılır. |
2 | yapı Bu işlev, defstruct işlemi tarafından oluşturulan tipte bir yapı nesnesini tanımlamak için kullanılır. |
3 | struct-map Bu işlev, yapıdaki hangi anahtarlara hangi değerlerin atanacağını açıkça tanımlayarak anahtar değerlerine özel olarak değerler atamak için kullanılır. |
4 | Bireysel Alanlara Erişim Yapının bireysel alanlarına, yapı nesnesi ile birlikte anahtarlara erişilerek erişilebilir. |
5 | Değişmez Doğa Varsayılan olarak yapılar da değişmezdir, bu nedenle belirli bir anahtarın değerini değiştirmeye çalışırsak, bu değişmeyecektir. |
6 | Yapıya Yeni Bir Anahtar Ekleme Yapılar değişmez olduğundan, yapıya başka bir anahtarın eklenmesinin tek yolu yeni bir yapının oluşturulmasıdır. Bunun nasıl başarılabileceğine dair bir örnek aşağıdaki programda gösterilmektedir. |
Birçok kez belirtildiği gibi, Clojure, veri türlerinin çoğunun değişmez olduğu bir programlama dilidir; bu, bir değişkenin değerini değiştirmenin tek yolunun yeni bir değişken oluşturmak ve ona yeni bir değer atamak olduğu anlamına gelir. Bununla birlikte, Clojure değişken bir durum yaratabilen bazı öğeler sağlar. Bunun atom veri türü ile elde edilebileceğini gördük. Bunun diğer bir yolu da Temsilciler aracılığıyladır.
Agentsbireysel konumların bağımsız, asenkron değişimini sağlar. Aracılar, ömürleri boyunca tek bir depolama konumuna bağlıdır ve bir eylemin sonucu olarak yalnızca bu konumun (yeni bir duruma) mutasyonunun gerçekleşmesine izin verir. Eylemler, bir Aracının durumuna eşzamansız olarak uygulanan ve dönüş değeri Aracının yeni durumu olan işlevlerdir (isteğe bağlı olarak ek bağımsız değişkenlerle).
Ajanlar ile ilgili olarak Clojure'da aşağıdaki işlemler mümkündür.
Sr.No. | İşlemler ve Açıklama |
---|---|
1 | ajan Ajan komutu kullanılarak bir ajan oluşturulur. |
2 | göndermek Bu işlev, temsilciye bir değer göndermek için kullanılır. |
3 | kapatma ajanları Bu işlev, çalışan herhangi bir ajanı kapatmak için kullanılır. |
4 | göndermek Bir ajana, doğası gereği bloke eden bir fonksiyonun atandığı durumlar vardır. |
5 | beklemek Bir ajanın değeri güncellendiğinde bir gecikme olduğu için Clojure, ajanın güncellenmesini bekleyeceği zamanı milisaniye cinsinden belirtmek için kullanılan bir 'bekleme' işlevi sağlamıştır. |
6 | beklemek Geçerli iş parçacığını (süresiz olarak!), Bu iş parçacığından veya aracıdan aracı (lar) a şimdiye kadar gönderilen tüm eylemler gerçekleşene kadar engeller. Başarısız temsilciler üzerinde engellenecek. |
7 | ajan hatası Aracı başarısız olursa, aracının zaman uyumsuz eylemi sırasında atılan istisnayı döndürür. Aracı başarısız olmazsa sıfır döndürür. |
WatchersDeğişken türünün bir değeri değiştiğinde çağrılan atomlar ve referans değişkenler gibi değişken türlerine eklenen işlevlerdir. Örneğin, çağıran program bir atom değişkeninin değerini değiştirirse ve atom değişkenine bir izleyici işlevi eklenirse, atomun değeri değiştirilir değiştirilmez işlev başlatılacaktır.
Clojure for Watchers'da aşağıdaki işlevler mevcuttur.
izle ekle
Bir ajan / atom / var / ref referansına bir izleme işlevi ekler. Saat‘fn’4 bağımsız değişkenin "fn" si olmalıdır: bir anahtar, başvuru, eski durumu, yeni durumu. Referansın durumu değiştirildiğinde, kayıtlı saatlerin işlevleri çağrılacaktır.
Sözdizimi
Sözdizimi aşağıdadır.
(add-watch variable :watcher
(fn [key variable-type old-state new-state]))
Parameters- 'değişken', atomun veya referans değişkeninin adıdır. 'Değişken türü', atom veya referans değişken gibi değişken türüdür. 'eski durum ve yeni durum', değişkenin eski ve yeni değerini otomatik olarak tutacak parametrelerdir. 'Anahtar' referans başına benzersiz olmalıdır ve saati kaldır-kaldır ile çıkarmak için kullanılabilir.
Return Value - Yok.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns clojure.examples.example
(:gen-class))
(defn Example []
(def x (atom 0))
(add-watch x :watcher
(fn [key atom old-state new-state]
(println "The value of the atom has been changed")
(println "old-state" old-state)
(println "new-state" new-state)))
(reset! x 2))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
The value of the atom has been changed
old-state 0
new-state 2
izle-kaldır
Bir referans değişkenine eklenmiş bir saati kaldırır.
Sözdizimi
Sözdizimi aşağıdadır.
(remove-watch variable watchname)
Parameters- 'değişken', atomun veya referans değişkeninin adıdır. 'saat adı', izleme işlevi tanımlandığında saate verilen addır.
Return Value - Yok.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns clojure.examples.example
(:gen-class))
(defn Example []
(def x (atom 0))
(add-watch x :watcher
(fn [key atom old-state new-state]
(println "The value of the atom has been changed")
(println "old-state" old-state)
(println "new-state" new-state)))
(reset! x 2)
(remove-watch x :watcher)
(reset! x 4))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
The value of the atom has been changed
old-state 0
new-state 2
Yukarıdaki programdan, ikinci sıfırlama komutunun, izleyici listesinden kaldırıldığı için izleyiciyi tetiklemediğini açıkça görebilirsiniz.
Herhangi bir dilde Macrossatır içi kod oluşturmak için kullanılır. Clojure bir istisna değildir ve geliştiriciler için basit makro olanaklar sağlar. Makrolar, geliştiriciye dili geliştiricinin ihtiyaçlarına göre uyarlamak için güçlü bir yol sağlayan kod oluşturma rutinlerini yazmak için kullanılır.
Makrolar için kullanılabilen yöntemler aşağıdadır.
defmacro
Bu işlev, makronuzu tanımlamak için kullanılır. Makronun bir makro adı, bir parametre listesi ve makronun gövdesi olacaktır.
Sözdizimi
Sözdizimi aşağıdadır.
(defmacro name [params*] body)
Parameters- 'ad' makronun adıdır. 'parametreler', makroya atanan parametrelerdir. "vücut", makronun gövdesidir.
Return Value - Yok.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns clojure.examples.example
(:gen-class))
(defn Example []
(defmacro Simple []
(println "Hello"))
(macroexpand '(Simple)))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
Hello
Yukarıdaki programdan, 'Basit' makrosunun satır içi olarak 'println' "Merhaba" olarak genişletildiğini görebilirsiniz. Makrolar, bir forma yönelik bağımsız değişkenlerin makrolar durumunda değerlendirilmesinin tek farkı ile işlevlere benzer.
makro genişletme
Bu, bir makroyu genişletmek ve kodu programa satır içi yerleştirmek için kullanılır.
Sözdizimi
Sözdizimi aşağıdadır.
(macroexpand macroname)
Parameters - 'makro adı', genişletilmesi gereken makronun adıdır.
Return Value - Genişletilmiş makro.
Misal
Bunun nasıl kullanıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
(ns clojure.examples.example
(:gen-class))
(defn Example []
(defmacro Simple []
(println "Hello"))
(macroexpand '(Simple)))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
Hello
Bağımsız Değişkenlerle Makro
Makrolar, argümanları almak için de kullanılabilir. Makro, herhangi bir sayıda argüman alabilir. Aşağıdaki örnek, argümanların nasıl kullanılabileceğini göstermektedir.
Misal
(ns clojure.examples.example
(:gen-class))
(defn Example []
(defmacro Simple [arg]
(list 2 arg))
(println (macroexpand '(Simple 2))))
(Example)
Yukarıdaki örnek, Basit makroya bir argüman yerleştirir ve ardından argümanı bir listeye argüman değeri eklemek için kullanır.
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
(2 2)
Reference valuesClojure'un değişken değişkenlere sahip olma talebiyle çalışmasının başka bir yoludur. Clojure, atomlar, ajanlar ve referans türleri gibi değişken veri türleri sağlar.
Aşağıda referans değerler için mevcut işlemler verilmiştir.
Sr.No. | İşlemler ve Açıklama |
---|---|
1 | ref Bu, bir referans değeri oluşturmak için kullanılır. Bir referans değeri oluştururken, oluşturulan değeri doğrulayacak bir doğrulayıcı işlevi sağlama seçeneği vardır. |
2 | ref-set Bu işlev, eski değer ne olursa olsun bir referansın değerini yeni bir değere ayarlamak için kullanılır. |
3 | değiştirmek Bu işlev, bir referans türünün değerini güvenli bir şekilde değiştirmek için kullanılır. Bu, başka bir işlem tarafından erişilemeyen bir iş parçacığında çalıştırılır. |
4 | 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. |
5 | işe gidip gelmek İşe gidip gelme, tıpkı alter ve ref-set gibi bir referans türünün değerini değiştirmek için de kullanılır. |
Veritabanı işlevini kullanmak için lütfen önce jdbc files şu url'den - https://codeload.github.com/clojure/java.jdbc/zip/master
Clojure'un veritabanlarına bağlanabilmesi için gerekli sürücüleri içeren bir zip dosyası bulacaksınız. Zip dosyası çıkarıldıktan sonra, sıkıştırılmamış konumu sınıf yolunuza eklediğinizden emin olun.
Veritabanı bağlantısı için ana dosya, jdbc.clj lokasyon clojure / java.
Clojure jdbc konektörü, bazıları aşağıdaki gibi çok çeşitli veritabanlarını destekler.
- H2Database
- Oracle
- Microsoft SQL Sunucusu
- MySQL
- PostgreSQL
Örneğimizde, örnek olarak MySQL DB kullanacağız.
Clojure'da Veritabanları ile ilgili olarak aşağıdaki işlemler mümkündür.
Veritabanı Bağlantısı
Bir MySQL veritabanına bağlanmadan önce aşağıdakilerden emin olun -
Bir TESTDB veritabanı oluşturdunuz.
TESTDB'de EMPLOYEE tablosu oluşturdunuz.
Bu tabloda FIRST_NAME, LAST_NAME, AGE, SEX ve INCOME alanları var.
"Testuser" kullanıcı kimliği ve "test123" şifresi TESTDB'ye erişmek için ayarlanmıştır.
'Mysql jar dosyasını' indirdiğinizden ve dosyayı sınıf yolunuza eklediğinizden emin olun.
MySQL Temellerini anlamak için MySQL eğitiminden geçtiniz .
Sözdizimi
Clojure'da bir bağlantı oluşturmak için sözdizimi aşağıdadır.
(def connection_name {
:subprotocol “protocol_name”
:subname “Location of mysql DB”
:user “username” :password “password” })
Parameters- 'bağlantı_adı' bağlantıya verilecek isimdir. "alt protokol", bağlantı için kullanılacak protokoldür. Varsayılan olarak mysql protokolünü kullanacağız. 'subname', veritabanı adı ile birlikte mysql db'ye bağlanmak için kullanılan url'dir. 'kullanıcı', veritabanına bağlanmak için kullanılan kullanıcı adıdır. 'şifre', veri tabanına bağlanmak için kullanılacak şifredir.
Return Value - Bu, sonraki mysql işlemlerinde kullanılabilecek bir bağlantı dizesi sağlayacaktır.
Aşağıdaki örnek, bilgi şemasındaki tablolara nasıl bağlanılacağını ve tablodaki tüm verilerin nasıl alınacağını gösterir.
Misal
(ns test.core
(:require [clojure.java.jdbc :as sql]))
(defn -main []
(def mysql-db {
:subprotocol "mysql"
:subname "//127.0.0.1:3306/information_schema"
:user "root"
:password "shakinstev"})
(println (sql/query mysql-db
["select table_name from tables"]
:row-fn :table_name)))
Verileri Sorgulama
Herhangi bir veritabanında veri sorgulamak, veritabanından bazı yararlı bilgileri almak anlamına gelir. Bir veritabanı bağlantısı kurulduktan sonra, bu veritabanına bir sorgulama yapmaya hazırsınız. Aşağıda, verilerin Clojure kullanılarak sorgulanabileceği sözdizimi verilmiştir.
Sözdizimi
clojure.java.jdbc/query dbconn
["query"]
:row-fn :sequence
Parameters- 'dbconn', veritabanına bağlanmak için kullanılan bağlantının adıdır. 'sorgu', veritabanından veri almak için kullanılan sorgu dizesidir. ': sıra' varsayılan olarak veritabanından getirilen tüm veri satırlarıdır ve bir sıra olarak döndürülür. Daha sonra hangi verilerin getirildiğini görmek için sıra üzerinde gerekli işlemler yapılabilir.
Return Value - Bu, sorgu işleminden veri satırlarını içeren bir sıra döndürür.
Aşağıdaki örnek, çalışan tablosuna nasıl bağlanılacağını ve tablodaki satırların ilk_adı sütununun nasıl alınacağını gösterir.
Misal
(ns test.core
(:require [clojure.java.jdbc :as sql]))
(defn -main []
(def mysql-db {
:subprotocol "mysql"
:subname "//127.0.0.1:3306/testdb"
:user "root"
:password "shakinstev"})
(println (sql/query mysql-db
["select first_name from employee"]
:row-fn :first_name)))
Yukarıdaki koddan bunu görebiliriz
Sorgu dizesi olarak “çalışandan ilk_adı seç” sorgusu geçilir.
İlk_adı, getirme işleminin bir sonucu olarak döndürülen dizidir.
Veritabanımızda John'un bir first_name değerini içeren tek bir satır olduğunu varsayarsak, aşağıdaki programın çıktısı aşağıdaki olacaktır.
(John)
Veri Ekleme
Kayıtlarınızı bir veritabanı tablosunda oluşturmak istediğinizde gereklidir. Aşağıda Clojure kullanılarak verilerin eklenebileceği sözdizimi verilmiştir. Bu,‘insert!’ işlevi.
Sözdizimi
clojure.java.jdbc/insert!
:table_name {:column_namen columnvalue}
Parameters- ': tablo_adı', eklemenin yapılması gereken tablonun adıdır. '{: column_namen columnvalue}', tabloda bir satır olarak eklenmesi gereken tüm sütun adlarının ve değerlerinin bir haritasıdır.
Return Value - Ekleme başarıyla yapılırsa bu sıfır döndürür.
Aşağıdaki örnek, testdb veritabanındaki çalışan tablosuna nasıl kayıt ekleneceğini gösterir.
Misal
(ns test.core
(:require [clojure.java.jdbc :as sql]))
(defn -main []
(def mysql-db {
:subprotocol "mysql"
:subname "//127.0.0.1:3306/testdb"
:user "root"
:password "shakinstev"})
(sql/insert! mysql-db
:employee {:first_name "John" :last_name "Mark" :sex "M" :age 30 :income 30}))
Şimdi MySQL veritabanınızı ve çalışan tablosunu kontrol ederseniz, yukarıdaki satırın tabloya başarıyla ekleneceğini göreceksiniz.
Verileri Silme
Bir tablodan satırlar silinebilir. ‘delete!’işlevi. Bu işlemin nasıl gerçekleştirilebileceğine dair sözdizimi aşağıdadır.
Sözdizimi
clojure.java.jdbc/delete!
:table_name [condition]
Parameters- ': tablo_adı', eklemenin yapılması gereken tablonun adıdır. "koşul", tablodan hangi satırın silinmesi gerektiğini belirlemek için kullanılan koşuldur.
Return Value - Bu, silinen satır sayısını döndürür.
Aşağıdaki örnek, testdb veritabanındaki çalışan tablosundan bir kaydın nasıl silineceğini gösterir. Örnek, yaşın 30'a eşit olması koşuluna göre tablodan bir satırı siler.
Misal
(ns test.core
(:require [clojure.java.jdbc :as sql]))
(defn -main []
(def mysql-db {
:subprotocol "mysql"
:subname "//127.0.0.1:3306/testdb"
:user "root"
:password "shakinstev"})
(println (sql/delete! mysql-db
:employee ["age = ? " 30])))
Yaşı 30 değerine eşit bir satıra sahip bir kaydınız varsa, bu satır silinecektir.
Verileri Güncelleme
Bir tablodaki satırlar, ‘update!’işlevi. Bu işlemin nasıl gerçekleştirilebileceğine dair sözdizimi aşağıdadır.
Sözdizimi
clojure.java.jdbc/update!
:table_name
{setcondition}
[condition]
Parameters- ': tablo_adı', eklemenin yapılması gereken tablonun adıdır. 'setcondition', bir harita açısından belirtildiği gibi güncellenmesi gereken sütundur. "koşul", tablodan hangi satırın silinmesi gerektiğini belirlemek için kullanılan koşuldur.
Return Value - Bu, güncellenen satır sayısını döndürür.
Aşağıdaki örnek, testdb veritabanındaki çalışan tablosundan bir kaydın nasıl silineceğini gösterir. Örnek, yaşın 30'a eşit olması koşuluna göre tablodan bir satırı günceller ve gelir değerini 40 olarak günceller.
(ns test.core
(:require [clojure.java.jdbc :as sql]))
(defn -main []
(def mysql-db {
:subprotocol "mysql"
:subname "//127.0.0.1:3306/testdb"
:user "root"
:password "shakinstev"})
(println (sql/update! mysql-db
:employee
{:income 40}
["age = ? " 30])))
Yaşı 30 değerine eşit bir satıra sahip bir kaydınız varsa, bu satır güncellenecek ve burada gelir değeri 40 olarak ayarlanacaktır.
İşlemler
İşlemler, veri tutarlılığını sağlayan mekanizmalardır. İşlemler aşağıdaki dört özelliğe sahiptir -
Atomicity - Ya bir işlem tamamlanır ya da hiçbir şey olmaz.
Consistency - Bir işlem tutarlı bir durumda başlamalı ve sistemi tutarlı bir durumda bırakmalıdır.
Isolation - Bir işlemin ara sonuçları cari işlemin dışında görünmez.
Durability - Bir işlem yapıldığında, sistem hatasından sonra bile etkiler kalıcıdır.
Misal
Aşağıdaki örnek Clojure'da işlemlerin nasıl uygulanacağını gösterir. Bir işlemde gerçekleştirilmesi gereken herhangi bir işlemin‘with-dbtransaction’ fıkra.
(ns test.core
(:require [clojure.java.jdbc :as sql]))
(defn -main []
(def mysql-db {
:subprotocol "mysql"
:subname "//127.0.0.1:3306/testdb"
:user "root"
:password "shakinstev"})
(sql/with-db-transaction [t-con mysql-db]
(sql/update! t-con
:employee
{:income 40}
["age = ? " 30])))
Bildiğimiz gibi Clojure kodu sonunda Java sanal ortamında çalışır. Bu nedenle Clojure'un Java'nın tüm işlevlerini kullanabilmesi mantıklıdır. Bu bölümde Clojure ve Java arasındaki ilişkiyi tartışalım.
Java Yöntemlerini Çağırma
Java yöntemleri nokta notasyonu kullanılarak çağrılabilir. Bir örnek dizelerdir. Clojure'daki tüm dizeler zaten Java dizgeleri olduğundan, dizeler üzerinde normal Java yöntemlerini çağırabilirsiniz.
Bunun nasıl yapıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
Misal
(ns Project
(:gen-class))
(defn Example []
(println (.toUpperCase "Hello World")))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir. Koddan, herhangi bir dize yöntemi için sadece nokta gösterimini çağırırsanız, Clojure'da da çalışacağını görebilirsiniz.
Çıktı
HELLO WORLD
Java Yöntemlerini Parametrelerle Çağırma
Java yöntemlerini parametrelerle de çağırabilirsiniz. Bunun nasıl yapıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
Misal
(ns Project
(:gen-class))
(defn Example []
(println (.indexOf "Hello World","e")))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir. Yukarıdaki koddan "e" parametresini indexOf metoduna geçirdiğimizi görebilirsiniz. Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
1
Java Nesneleri Oluşturma
Clojure'da, Java'da yapılana benzer şekilde 'new' anahtar kelimesi kullanılarak nesneler oluşturulabilir.
Bunun nasıl yapıldığına dair bir örnek aşağıdaki programda gösterilmektedir.
Misal
(ns Project
(:gen-class))
(defn Example []
(def str1 (new String "Hello"))
(println str1))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir. Java'dan mevcut String sınıfından yeni bir nesne oluşturmak için 'new' anahtar kelimesini kullanabileceğimizi yukarıdaki koddan görebilirsiniz. Java'da yaptığımız gibi, nesneyi oluştururken değeri aktarabiliriz. Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
Hello
Aşağıda, Integer sınıfının bir nesnesini nasıl yaratabileceğimizi ve bunları normal Clojure komutlarında nasıl kullanabileceğimizi gösteren başka bir örnek verilmiştir.
Misal
(ns Project
(:gen-class))
(defn Example []
(def my-int(new Integer 1))
(println (+ 2 my-int)))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
3
İçe Aktar Komutu
Java kitaplıklarını ad alanına dahil etmek için import komutunu da kullanabiliriz, böylece sınıflara ve yöntemlere kolayca erişilebilir.
Aşağıdaki örnek, import komutunu nasıl kullanabileceğimizi göstermektedir. Örnekte, sınıfları içe aktarmak için import komutunu kullanıyoruz.java.util.stackkütüphane. Daha sonra stack sınıfının push ve pop yöntemini olduğu gibi kullanabiliriz.
Misal
(ns Project
(:gen-class))
(import java.util.Stack)
(defn Example []
(let [stack (Stack.)]
(.push stack "First Element")
(.push stack "Second Element")
(println (first stack))))
(Example)
Yukarıdaki program aşağıdaki çıktıyı üretir.
Çıktı
First Element
Java Komutunu Kullanarak Kod Çalıştırma
Clojure kodu Java komutu kullanılarak çalıştırılabilir. Bunun nasıl yapılacağına dair sözdizimi aşağıdadır.
java -jar clojure-1.2.0.jar -i main.clj
Tüm Clojure tabanlı sınıfların JVM'ye yüklenebilmesi için Clojure jar dosyasını belirtmelisiniz. 'Main.clj' dosyası, yürütülmesi gereken Clojure kod dosyasıdır.
Yerleşik Java İşlevleri
Clojure, Java'nın yerleşik işlevlerinin çoğunu kullanabilir. Bazıları -
Math PI function- Clojure, Math yöntemini PI değerine kadar kullanabilir. Aşağıda örnek bir kod verilmiştir.
Misal
(ns Project
(:gen-class))
(defn Example []
(println (. Math PI)))
(Example)
Yukarıdaki kod aşağıdaki çıktıyı üretir.
Çıktı
3.141592653589793
System Properties- Clojure, sistem özelliklerini de sorgulayabilir. Aşağıda örnek bir kod verilmiştir.
Misal
(ns Project
(:gen-class))
(defn Example []
(println (.. System getProperties (get "java.version"))))
(Example)
Sistemdeki Java sürümüne bağlı olarak, karşılık gelen değer görüntülenecektir. Aşağıda örnek bir çıktı verilmiştir.
Çıktı
1.8.0_45
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. Yoluyla açığa çıkan dinamik var sistemi, iş parçacıkları içindeki değişen durumu izole etmeyi 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.
Daha önce dosync bloğunun ne yaptığını 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ştirmemiz gerekir.
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
Clojure, oluşturmaya olanak sağlayan bazı kütüphanelere sahiptir. Desktop ve Web-based applications. Her birini tartışalım.
Sr.No. | Uygulamalar ve Açıklama |
---|---|
1 | Masaüstü - Testere See-saw, masaüstü uygulamaları oluşturmak için kullanılabilecek bir kitaplıktır. |
2 | Masaüstü - Metnin Değerini Değiştirme Penceredeki içeriğin değeri, kullanılarak değiştirilebilir. ‘config!’seçeneği. Aşağıdaki örnekte config! seçeneği, pencere içeriğini yeni “Hoşçakal” değerine değiştirmek için kullanılır. |
3 | Masaüstü - Kalıcı İletişim Kutusu Görüntüleme See-saw sınıfının uyarı yöntemi kullanılarak kalıcı bir iletişim kutusu gösterilebilir. Yöntem, kalıcı iletişim kutusunda gösterilmesi gereken metin değerini alır. |
4 | Masaüstü - Düğmelerin Görüntülenmesi Düğme sınıfı yardımıyla düğmeler görüntülenebilir. |
5 | Masaüstü - Etiketleri Görüntüleme Etiket sınıfı yardımıyla etiketler görüntülenebilir. |
6 | Masaüstü - Metin Alanlarını Görüntüleme Metin Alanları, metin sınıfı yardımıyla görüntülenebilir. |
Web Uygulamaları - Giriş
Clojure'da bir web uygulaması oluşturmak için aşağıdaki bağlantıda bulunan Ring uygulama kitaplığını kullanmanız gerekir. https://github.com/ring-clojure/ring
Siteden gerekli kavanozları indirdiğinizden ve Clojure uygulamasına bağımlılık olarak eklediğinizden emin olmanız gerekir.
Ring framework aşağıdaki yetenekleri sağlar -
İşleri, bir http isteği web uygulamanıza normal bir Clojure HashMap olarak gelecek şekilde ayarlar ve aynı şekilde bir HashMap olarak bir yanıt döndürebilmeniz için yapar.
Bu istek ve yanıt haritalarının tam olarak neye benzemesi gerektiğini açıklayan bir şartname sağlar.
Bir web sunucusu (Jetty) getirir ve web uygulamanızı ona bağlar.
Ring çerçevesi otomatik olarak bir web sunucusu başlatabilir ve Clojure uygulamasının bu sunucuda çalışmasını sağlar. O zaman Compojure çerçevesi de kullanılabilir. Bu, günümüzde çoğu modern web uygulamasının geliştirilme şekli olan rotaların oluşturulmasına izin verir.
Creating your first Clojure application - Aşağıdaki örnek, Clojure'da ilk web uygulamanızı nasıl oluşturabileceğinizi gösterir.
(ns my-webapp.handler
(:require [compojure.core :refer :all]
[compojure.route :as route]
[ring.middleware.defaults :refer [wrap-defaults site-defaults]]))
(defroutes app-routes
(GET "/" [] "Hello World")
(route/not-found "Not Found"))
(def app
(wrap-defaults app-routes site-defaults))
Programın aşağıdaki yönlerine bakalım -
‘defroutes’ Web uygulamasına farklı rotalara yapılan talep Clojure uygulamanızdaki farklı işlevlere yönlendirilebilmesi için rotalar oluşturmak için kullanılır.
Yukarıdaki örnekte, "/" varsayılan yol olarak bilinir, bu nedenle web uygulamanızın tabanına göz attığınızda, "Merhaba Dünya" dizesi web tarayıcısına gönderilecektir.
Kullanıcı, Clojure uygulaması tarafından işlenemeyen herhangi bir url'ye ulaşırsa, "Bulunamadı" dizesini gösterecektir.
Clojure uygulamasını çalıştırdığınızda, uygulamanız varsayılan olarak localhost: 3000 olarak yüklenecektir, bu nedenle bu konuma göz atarsanız aşağıdaki çıktıyı alırsınız.
Web Uygulamaları - Web Uygulamanıza Daha Fazla Yol Ekleme
Web uygulamanıza daha fazla rota da ekleyebilirsiniz. Aşağıdaki örnek, bunun nasıl başarılacağını göstermektedir.
(ns my-webapp.handler
(:require [compojure.core :refer :all]
[compojure.route :as route]
[ring.middleware.defaults :refer [wrap-defaults site-defaults]]))
(defroutes app-routes
(GET "/" [] "Hello World")
(GET "/Tutorial" [] "This is a tutorial on Clojure")
(route/not-found "Not Found"))
(def app
(wrap-defaults app-routes site-defaults))
Uygulamaya bir rota eklemenin, url rotasıyla başka bir GET işlevi eklemek kadar kolay olduğunu görebilirsiniz. (GET "/ Tutorial" [] "Bu Clojure hakkında bir eğiticidir")
Konuma göz atarsanız http://localhost:3000/Tutorial, aşağıdaki çıktıyı alacaksınız.
Bu bölümde, Clojure tarafından sağlanan otomatik test seçeneklerini tartışalım.
İstemci Uygulamaları için Test
Clojure çerçevesi için testi kullanmak için, şu adreste bulunan bağımlılıkları kullanmanız gerekir: https://github.com/slagyr/speclj#manual-installation
Bu URL, specljClojure için Test verilerine dayalı veya Davranış odaklı bir test çerçevesi olarak kullanılan çerçeve. 'Speclj' kitaplıklarından herhangi birini kullanırken Clojure 1.7.0 çerçevesini kullandığınızdan emin olmalısınız. Varsayılan olarak, test dosyaları Clojure kod dosyalarından farklı olacaktır ve bir 'spec' dizinine yerleştirilmeleri gerekir.
Aşağıda bir test dosyası için örnek bir kod verilmiştir.
(ns change.core-spec
(:require [speclj.core :refer :all]))
(describe "Truth"
(it "is true"
(should true))
(it "is not false"
(should-not false)))
(run-specs)
Yukarıdaki kodla ilgili aşağıdaki hususlara dikkat edilmesi gerekir -
Öncelikle, 'speclj' çerçevesine tüm çekirdek kitaplıkları dahil etmek için 'require' ifadesini kullandığımızdan emin olmalıyız.
Sıradaki 'tanımla' işlevi. Bu, yaratılan test senaryosuna bir açıklama sağlamak için kullanılır.
Sonraki işlev, gerçek test durumu olan 'it' işlevidir. İlk test durumunda, "doğrudur" dizesi test senaryosuna verilen addır.
Olmalı ve olmamalı olarak bilinmelidir assertions. Tüm iddialar should ile başlar. Olmalı ve olmamalı, mevcut birçok iddiadan sadece ikisi. Her ikisi de sırasıyla doğruluk ve yanlışlığı kontrol edeceklerini ifade eder.
Test senaryosunu çalıştırırsanız, aşağıdaki çıktıyı alırsınız. Çıktı, test senaryosunun çalışması için milisaniye cinsinden geçen süreyi gösterir.
←[32m.←[0m←[32m.←[0m
Finished in 0.00014 seconds
Web Tabanlı Uygulamaları Test Etme
Seleniumgünümüzün web tabanlı uygulamalarını test etmek için kullanılan temel çerçevelerden biridir. Web tabanlı uygulamaları test etmek için kullanılabilecek Clojure kitaplıkları da mevcuttur.
Clojure web tabanlı uygulamaları test etmek için Selenium kitaplıklarını nasıl kullanabileceğimize bakalım.
Step 1- İlk adım, test edilmesi gereken web tabanlı bir uygulama oluşturmak için Ring ve Compojure çerçevesini kullandığımızdan emin olmaktır. Önceki bölümlerimizden örneklerden birini kullanalım. Aşağıdaki kod, tarayıcıda "Merhaba Dünya" görüntüleyen basit bir web uygulamasıdır.
(ns my-webapp.handler
(:require [compojure.core :refer :all]
[compojure.route :as route]
[ring.middleware.defaults :refer [wrap-defaults site-defaults]]))
(defroutes app-routes
(GET "/" [] "Hello World")
(route/not-found "Not Found"))
(def app
(wrap-defaults app-routes site-defaults))
Step 2 - Sonra selenyum kavanoz dosyasını indirdiğinizden emin olun https://mvnrepository.com/artifact/org.seleniumhq.selenium/selenium-server/2.47.0 ve sınıf yolunuza dahil edin.
Step 3 - Ayrıca aşağıdaki konumdan web testini çalıştırmak için kullanılacak olan 'clj' web sürücüsünü indirdiğinizden emin olun.
https://clojars.org/clj-webdriver/versions/0.7.1
Step 4 - Proje dizininizde, özellikler adlı başka bir dizin oluşturun ve 'config.clj' adlı bir dosya oluşturun.
Step 5 - Daha sonra, önceki adımda oluşturulan 'config.clj' dosyasına aşağıdaki kodu ekleyin.
ns clj-webdriver-tutorial.features.config)
(def test-port 3000)
(def test-host "localhost")
(def test-base-url (str "http://" test-host ":" test-port "/"))
Yukarıdaki kod, temel olarak web test çerçevesine, URL'ye yüklenen uygulamayı test etmesini söyler. http://localhost:3000
Step 6 - Son olarak, testimizi gerçekleştirmek için kodumuzu yazalım.
(ns clj-webdriver-tutorial.features.homepage
(:require [clojure.test :refer :all]
[ring.adapter.jetty :refer [run-jetty]]
[clj-webdriver.taxi :refer :all]
[clj-webdriver-tutorial.features.config :refer :all]
[clj-webdriver-tutorial.handler :refer [app-routes]]))
(ns clj-webdriver-tutorial.features.homepage
(:require [clojure.test :refer :all]
[ring.adapter.jetty :refer [run-jetty]]
[clj-webdriver.taxi :refer :all]
[clj-webdriver-tutorial.features.config :refer :all]
[clj-webdriver-tutorial.handler :refer [app-routes]]))
(defn start-server []
(loop [server (run-jetty app-routes {:port test-port, :join? false})]
(if (.isStarted server)
server
(recur server))))
(defn stop-server [server]
(.stop server))
(defn start-browser []
(set-driver! {:browser :firefox}))
(defn stop-browser []
(quit))
(deftest homepage-greeting
(let [server (start-server)]
(start-browser)
(to test-base-url)
(is (= (text "body") "Hello World"))
(stop-browser)
(stop-server server)))
Yukarıdaki kod aşağıdaki eylemleri gerçekleştirecektir -
- Uygulama için sunucuyu başlatın.
- Tarayıcıda kök yolunu açın.
- Sayfada "Merhaba Dünya" mesajının olup olmadığını kontrol edin.
- Tarayıcıyı kapatın.
- Sunucuyu kapatın.
Clojure kütüphanesini bu kadar güçlü kılan şey, Clojure çerçevesi için mevcut olan kütüphane sayısıdır. Web testi, web geliştirme, swing tabanlı uygulamalar geliştirme, MySQL veritabanlarına bağlanmak için jdbc kütüphanesi gibi önceki örneklerimizde kullanılan pek çok kitaplık gördük. Aşağıda birkaç kitaplıktan sadece birkaç örnek verilmiştir.
data.xml
Bu kütüphane Clojure'un XML verileriyle çalışmasına izin verir. Kullanılacak kitaplık sürümü org.clojure / data.xml "0.0.8" dir. Data.xml, XML'in ayrıştırılmasını ve yayılmasını destekler. Ayrıştırma işlevleri, bir Reader veya InputStream'den XML okuyacaktır.
Misal
Aşağıda, bir dizeden XML'e veri işlemenin bir örneği verilmiştir.
(ns clojure.examples.example
(use 'clojure.data.xml)
(:gen-class))
(defn Example []
(let [input-xml (java.io.StringReader. "<?xml version = \"1.0\"
encoding = \"UTF-8\"?><example><clo><Tutorial>The Tutorial
value</Tutorial></clo></example>")]
(parse input-xml)))
#clojure.data.xml.Element{
:tag :example, :attrs {}, :content (#clojure.data.xml.Element {
:tag :clo, :attrs {}, :content (#clojure.data.xml.Element {
:tag :Tutorial, :attrs {},:content ("The Tutorial value")})})}
(Example)
data.json
Bu kütüphane Clojure'un JSON verileriyle çalışmasına izin verir. Kullanılacak kitaplık sürümü org.clojure / data.json "0.2.6" dır.
Misal
Aşağıda bu kütüphanenin kullanımına bir örnek verilmiştir.
(ns clojure.examples.example
(:require [clojure.data.json :as json])
(:gen-class))
(defn Example []
(println (json/write-str {:a 1 :b 2})))
(Example)
Çıktı
Yukarıdaki program aşağıdaki çıktıyı üretir.
{\"a\":1,\"b\":2}
data.csv
Bu kütüphane Clojure'un ‘csv’veri. Kullanılacak kitaplık sürümü org.clojure / data.csv "0.1.3" dür.
Misal
Aşağıda bu kütüphanenin kullanımına bir örnek verilmiştir.
(ns clojure.examples.example
(require '[clojure.data.csv :as csv]
'[clojure.java.io :as io])
(:gen-class))
(defn Example []
(with-open [in-file (io/reader "in-file.csv")]
(doall
(csv/read-csv in-file)))
(with-open [out-file (io/writer "out-file.csv")]
(csv/write-csv out-file
[[":A" "a"]
[":B" "b"]])))
(Example)
Yukarıdaki kodda, 'csv' işlevi önce adlı bir dosyayı okuyacaktır. in-file.csvve tüm verileri dosyadaki değişkene koyun. Ardından, tüm verileri adlı bir dosyaya yazmak için write-csv işlevini kullanıyoruz.out-file.csv.